Advertisement
Guest User

Untitled

a guest
Jul 7th, 2012
353
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Lua 14.49 KB | None | 0 0
  1. --
  2. -- DESCRIPTION:  Maze generation program Lua <-> C:
  3. --               Lua - maze generation
  4. --               C   - maze visualization
  5. --      AUTHOR:  Alexander Simakov, <xdr [dot] box [at] Gmail>
  6. --               http://alexander-simakov.blogspot.com/
  7. --     LICENSE:  Public domain
  8. --  HOW-TO RUN:  gcc -o maze_generator -Wall `pkg-config lua5.1 --libs --cflags` maze_generator.c
  9. --              ./maze_generator ./maze_dfs.lua 15 10
  10. --
  11. --               OR
  12. --
  13. --               Uncomment usage example at the end of the file and
  14. --               run as a stand alone Lua program:
  15. --               chmod 755 maze_dfs.lua
  16. --               ./maze_dfs.lua
  17.  
  18. local Maze = {}
  19.  
  20. --
  21. -- Public methods
  22. --
  23.  
  24. -- Create new Maze instance
  25. function Maze:new(width, height)
  26.     local obj = {
  27.         width       = width,
  28.         height      = height,
  29.         cells       = {},
  30.         ascii_board = {},
  31.         visited = 0,
  32.         max = width*height,
  33.     }
  34.  
  35.     setmetatable(obj, self)
  36.     self.__index = self
  37.  
  38.     return obj
  39. end
  40.  
  41. -- Generate maze
  42. function Maze:generate()
  43.     self:_init_cells()
  44.     self:_init_ascii_board()
  45.  
  46.     self.cells[1][1].left_wall                     = false -- open entry point
  47.     self.cells[self.height][self.width].right_wall = false -- open exit point
  48.  
  49.     self:_process_neighbor_cells(1, 1);
  50.     local result = {self:_render()}
  51.     return unpack(result)
  52. end
  53.  
  54. --
  55. -- Private methods
  56. --
  57.  
  58. -- Close all walls, mark all cells as not visited yet
  59. function Maze:_init_cells()
  60.     self.cells = {}
  61.     for y = 1, self.height do
  62.         self.cells[y] = {}
  63.         for x = 1, self.width do
  64.             self.cells[y][x] = {
  65.                 left_wall   = true,
  66.                 right_wall  = true,
  67.                 top_wall    = true,
  68.                 bottom_wall = true,
  69.                 is_visited  = false,
  70.             }
  71.         end
  72.     end
  73.  
  74.     return
  75. end
  76.  
  77. -- Draw +---+---+ ... +---+
  78. function Maze:_draw_horizontal_bar()
  79.     local line = ''
  80.  
  81.     for x = 1, self.width do
  82.         line = line .. '+---'
  83.     end
  84.     line = line .. '+'
  85.  
  86.     return line
  87. end
  88.  
  89. -- Draw |   |   | ... |   |
  90. function Maze:_draw_vertical_bars()
  91.     local line = ''
  92.  
  93.     for x = 1, self.width do
  94.         line = line .. '|   '
  95.     end
  96.     line = line .. '|'
  97.  
  98.     return line
  99. end
  100.  
  101. -- Draw ascii chess-like board with all walls closed
  102. function Maze:_init_ascii_board()
  103.     self.ascii_board = {}
  104.  
  105.     for y = 1, self.height do
  106.         local horizontal_bar = self:_string_to_array(self:_draw_horizontal_bar());
  107.         local vertical_bars  = self:_string_to_array(self:_draw_vertical_bars());
  108.  
  109.         table.insert( self.ascii_board, horizontal_bar )
  110.         table.insert( self.ascii_board, vertical_bars )
  111.     end
  112.  
  113.     local horizontal_bar = self:_string_to_array(self:_draw_horizontal_bar());
  114.     table.insert( self.ascii_board, horizontal_bar )
  115.  
  116.     return
  117. end
  118.  
  119. -- Get cells neighbor to the cell (x,y): left, right, top, bottom
  120. function Maze:_get_neighbor_cells(x, y)
  121.     local neighbor_cells = {}
  122.     local shifts = {
  123.         { x = -1, y =  0 },
  124.         { x =  1, y =  0 },
  125.         { x =  0, y =  1 },
  126.         { x =  0, y = -1 },
  127.     }
  128.     for index, shift in ipairs(shifts) do
  129.         new_x = x + shift.x
  130.         new_y = y + shift.y
  131.         if new_x >= 1 and new_x <= self.width  and
  132.            new_y >= 1 and new_y <= self.height
  133.         then
  134.             table.insert( neighbor_cells, { x = new_x, y = new_y } )
  135.         end
  136.     end
  137.     return neighbor_cells
  138. end
  139. Num = 0
  140. -- Process the cell with all its neighbors in random order
  141.  
  142. function round(num, idp)
  143.   local mult = 10^(idp or 0)
  144.   return math.floor(num * mult + 0.5) / mult
  145. end
  146.  
  147. function Maze:_process_neighbor_cells(ox, oy)
  148.     local cellList = { {ox,oy} }
  149.     local Prev = 0
  150.     while #cellList > 0 do
  151.         local cell = table.remove(cellList,1)
  152.         local x,y = cell[1],cell[2]
  153.         local parent = cell[3]
  154.         if not self.cells[y][x].is_visited then
  155.             self.cells[y][x].is_visited = true
  156.             self.visited = self.visited+1
  157.             if Prev ~= round(((self.visited)/(self.max)*100)) then
  158.               Prev = round(((self.visited)/(self.max)*100))
  159.               print(Prev.."%")
  160.               os.queueEvent("randomEvent")
  161.               os.pullEvent()
  162.             end
  163.             local neighbor_cells = self:_shuffle(self:_get_neighbor_cells(x, y))
  164.             if parent then
  165.                 if parent.x > x     then     -- open wall with right neighbor
  166.                     self.cells[y][x].right_wall                              = false
  167.                     self.cells[parent.y][parent.x].left_wall   = false
  168.                 elseif parent.x < x then     -- open wall with left neighbor
  169.                     self.cells[y][x].left_wall                               = false
  170.                     self.cells[parent.y][parent.x].right_wall  = false
  171.                 elseif parent.y > y then     -- open wall with bottom neighbor
  172.                     self.cells[y][x].bottom_wall                             = false
  173.                     self.cells[parent.y][parent.x].top_wall    = false
  174.                 elseif parent.y < y then     -- open wall with top neighbor
  175.                     self.cells[y][x].top_wall                                = false
  176.                     self.cells[parent.y][parent.x].bottom_wall = false
  177.                 end
  178.              
  179.             end
  180.             for index, neighbor_cell in ipairs(neighbor_cells) do
  181.                 if self.cells[neighbor_cell.y][neighbor_cell.x].is_visited == false then
  182.                     table.insert(cellList,1,{neighbor_cell.x,neighbor_cell.y,{x=x,y=y}})
  183.                 end
  184.                 --self:_process_neighbor_cells(neighbor_cell.x, neighbor_cell.y)
  185.             end
  186.         end
  187.     end
  188.  
  189.     return
  190. end
  191.  
  192.  
  193. function Maze:_wipe_left_wall(x, y)
  194.     self.ascii_board[y * 2][(x - 1) * 4 + 1] = ' '
  195.     return
  196. end
  197.  
  198. function Maze:_wipe_right_wall(x, y)
  199.     self.ascii_board[y * 2][x * 4 + 1] = ' '
  200.     return
  201. end
  202.  
  203. function Maze:_wipe_top_wall(x, y)
  204.     for i = 0, 2 do
  205.         self.ascii_board[(y - 1) * 2 + 1][ (x - 1) * 4 + 2 + i] = ' '
  206.     end
  207.     return
  208. end
  209.  
  210. function Maze:_wipe_bottom_wall(x, y)
  211.     for i = 0, 2 do
  212.         self.ascii_board[y * 2 + 1][(x - 1) * 4 + 2 + i] = ' '
  213.     end
  214.     return
  215. end
  216.  
  217. function Maze:_render()
  218.     for y = 1, self.height do
  219.         for x = 1, self.width do
  220.             if self.cells[y][x].left_wall == false then
  221.                 self:_wipe_left_wall(x, y)
  222.             end
  223.  
  224.             if self.cells[y][x].right_wall == false then
  225.                 self:_wipe_right_wall(x, y)
  226.             end
  227.  
  228.             if self.cells[y][x].top_wall == false then
  229.                 self:_wipe_top_wall(x, y)
  230.             end
  231.  
  232.             if self.cells[y][x].bottom_wall == false then
  233.                 self:_wipe_bottom_wall(x, y)
  234.             end
  235.         end
  236.     end
  237.  
  238.     local result = {}
  239.     for index, chars in ipairs(self.ascii_board) do
  240.         result[index] = self:_array_to_string(chars)
  241.     end
  242.  
  243.     return unpack(result)
  244. end
  245.  
  246. --
  247. -- Utils
  248. --
  249.  
  250. -- Generate random number: use either external random
  251. -- number generator or internal - math.random()
  252. function Maze:_rand(max)
  253.     if type(external_rand) ~= "nil" then
  254.         return external_rand(max)
  255.     else
  256.         return math.random(max)
  257.     end
  258. end
  259.  
  260. -- Shuffle array (external_rand() is external C function)
  261. function Maze:_shuffle(array)
  262.     for i = 1, #array do
  263.         index1 = self:_rand(#array)
  264.         index2 = self:_rand(#array)
  265.  
  266.         if index1 ~= index2 then
  267.             array[index1], array[index2] = array[index2], array[index1]
  268.         end
  269.     end
  270.  
  271.     return array
  272. end
  273.  
  274. -- Split string into array of chars
  275. function Maze:_string_to_array(str)
  276.     local array = {}
  277.     for char in string.gmatch(str, '.') do
  278.         table.insert(array, char)
  279.     end
  280.  
  281.     return array
  282. end
  283.  
  284. function Maze:_array_to_string(array)
  285.     return table.concat(array, '')
  286. end
  287.  
  288. function generate_maze(width, height)
  289.     maze = Maze:new(width, height)
  290.     return maze:generate()
  291. end
  292. --[[
  293. math.randomseed( os.time() )
  294.  
  295. for i = 1, 10 do
  296.     print(generate_maze(6, 5))
  297.     os.execute('sleep 1')
  298. end
  299. --]]
  300.  
  301. --Map API. Revised.
  302. --[[
  303. Cheat sheet
  304. createMap(MapName)
  305. loadMap(MapName) -- Load a file into the API
  306. cView(Viewer, Map, xWide, yHigh) -- Create a view point on this map which is x chars wide and y chars high
  307. dView(Viewer) -- delete the viewer
  308. draw(Viewer) -- Draws the viewpoint from the viewer.
  309. editMap(Map, X, Y, Char) -- Edit the map at this point to replace the char there with this char
  310. returnView(Viewer) -- Returns the view to whatever called it. Catch like this Stuff = {returnView("ME!!")} then its loaded into the table "stuff" in your program
  311. moveScreen(Viewer, X, Y) Moves the viewpoint of the viewer to that x, y
  312. saveMap(Map, Location) Saves the loaded map into the location
  313. replaceLine(Map, LineNo, From, To, StuffToReplaceWith) replaceLine("World", 3, 10, 30, "") would replace line 3 at map World chars 10 to 30 with ""
  314. insertLine(Map, y) Makes a new line there
  315. removeLine(Map, y) do I really need to say this?
  316. ]]
  317. Maps = {}
  318. Views = {}
  319.  
  320. function replaceLine(Map, y, from, to, Stuff)
  321.   Maps[Map][y] = string.sub(Maps[Map][y], 1, from)..Stuff..string.sub(Maps[Map][y], to, string.len(Maps[Map][y]))
  322. end
  323.  
  324. function insertLine(Map, y, data)
  325.   table.insert(Maps[Map], y, data)
  326. end
  327.  
  328. function removeLine(Map, y)
  329.   table.remove(Maps[Map], y)
  330. end
  331.  
  332. function loadMap(MapName) -- Loads a map into a table, The info included is loaded into another table
  333.   if not fs.exists(MapName) then error("Please use a map that exists! Use something like Folder/Folder/File") end
  334.   Maps[MapName] = {}
  335.   local MapLoadVari = io.open(MapName, "r")
  336.   local i = 1 -- how many its done.
  337.   local a = 1 -- x len
  338.   local b = 1 -- y len
  339.   for line in MapLoadVari:lines() do
  340.     i = i+1
  341.     Maps[MapName][i] = line
  342.     b = b+1
  343.     if string.len(line) > a then
  344.         a = string.len(line)
  345.     end
  346.   end
  347.   Maps[MapName]["X"] = a
  348.   Maps[MapName]["Y"] = b
  349. end
  350.  
  351. function createMap(MapName)
  352.   Maps[MapName] = {}
  353.   Maps[MapName][1] = {}
  354. end
  355.  
  356. function cView(Viewer, Map, xWide, yHigh)
  357.   Views[Viewer] = {}
  358.   Views[Viewer]["Wide"] = xWide
  359.   Views[Viewer]["Map"] = Map
  360.   Views[Viewer]["High"] = yHigh
  361. end
  362.  
  363. function dView(Name)
  364.   Views[Viewer] = nil
  365. end
  366.  
  367. function draw(Viewer)
  368.   for n=1,Views[Viewer]["High"] do
  369.     if Maps[Views[Viewer]["Map"]][(Views[Viewer]["Y"]+n-1)] then
  370.        C = 0
  371.       if Views[Viewer]["X"] < 1 then
  372.         C = math.abs(Views[Viewer]["X"])
  373.       end
  374.       local Y = Views[Viewer]["Y"]
  375.       local X = Views[Viewer]["X"]
  376.       local B = (X+Views[Viewer]["Wide"])-string.len(Maps[(Views[Viewer]["Map"])][(Y+n-1)])
  377.       if B < 1 then
  378.         B = 0
  379.       end
  380.       Views[Viewer][n] = string.rep(" ", C)..string.sub(Maps[(Views[Viewer]["Map"])][(Y+n-1)], X+C+1, X+Views[Viewer]["Wide"])..string.rep(" ", B)
  381.     else
  382.       Views[Viewer][n] = string.rep(" ", Views[Viewer]["Wide"])
  383.     end
  384.   end
  385. end
  386.  
  387. function editMap(Map, X, Y, Char)
  388.   Maps[Map][Y] = string.sub(Maps[Map][Y], 1, X-1)..Char..string.sub(Maps[Map][Y], X+1, string.len(Maps[Map][Y]))
  389. end
  390.  
  391. function returnView(Viewer)
  392.   local Sec = {}
  393.   for n=1,Views[Viewer]["High"] do
  394.     Sec[n] = Views[Viewer][n]
  395.   end
  396.   return unpack(Sec)
  397. end
  398.  
  399. function returnChar(Map, x, y)
  400.   if x*y > 0 and Maps[Map][y] then
  401.     if string.sub(Maps[Map][y], x, x) ~= "" then
  402.       return string.sub(Maps[Map][y], x, x)
  403.     end
  404.   end
  405.   return false
  406. end
  407.  
  408. function moveScreen(Viewer, xAmount, yAmount) -- Was going to add in checking if bla bla exists but..
  409.   Views[Viewer]["X"] = xAmount
  410.   Views[Viewer]["Y"] = yAmount
  411. end
  412.  
  413. function saveMap(Map, Location)
  414.   local Beauty = io.open(Location, "w")
  415.   for n=1,#Maps[Map] do
  416.     Beauty:write(Maps[Map][n])
  417.     if #Maps[Map]+1 then
  418.       Beauty:write("\n")
  419.     end
  420.   end
  421.   Beauty:close()
  422. end
  423.  
  424. function getSize(Map)
  425.   return string.len(Maps[Map][1]), #Maps[Map]
  426. end
  427. --[[
  428. Cheat sheet
  429. createMap(MapName)
  430. loadMap(MapName) -- Load a file into the API
  431. cView(Viewer, Map, xWide, yHigh) -- Create a view point on this map which is x chars wide and y chars high
  432. dView(Viewer) -- delete the viewer
  433. draw(Viewer) -- Draws the viewpoint from the viewer.
  434. editMap(Map, X, Y, Char) -- Edit the map at this point to replace the char there with this char
  435. returnView(Viewer) -- Returns the view to whatever called it. Catch like this Stuff = {returnView("ME!!")} then its loaded into the table "stuff" in your program
  436. moveScreen(Viewer, X, Y) Moves the viewpoint of the viewer to that x, y
  437. saveMap(Map, Location) Saves the loaded map into the location
  438. replaceLine(Map, LineNo, From, To, StuffToReplaceWith) replaceLine("World", 3, 10, 30, "") would replace line 3 at map World chars 10 to 30 with ""
  439. insertLine(Map, y, data) Makes a new line there
  440. removeLine(Map, y) do I really need to say this?
  441. getSize(Map)
  442. returnChar(Map, X, Y) returns the char there
  443. ]]
  444.  
  445. Stuff = {...}
  446. if Stuff[1] and tonumber(Stuff[2]) then
  447.   Maps["Maze"] = {generate_maze(tonumber(Stuff[1]), tonumber(Stuff[2]))}
  448. else
  449.   Maps["Maze"] = {generate_maze(20, 20)}
  450. end
  451. if Stuff[3] then
  452.   math.randomseed(tonumber(Stuff[3]))
  453. end
  454. cView("Main", "Maze", term.getSize())
  455. local xx, yy = term.getSize("Maze")
  456. local x = 1
  457. local y = 2
  458. local Char = ">"
  459. local xx = math.floor(xx/2)
  460. local yy = math.floor(yy/2)
  461. moveScreen("Main", x-xx, y-yy)
  462. draw("Main")
  463. local G = {returnView("Main")}
  464. for n=1,#G do
  465.   term.setCursorPos(1,n)
  466.   term.write(G[n])
  467. end
  468. term.setCursorPos(xx, yy+1)
  469. term.write(Char)
  470. function safe(x, y)
  471.   print(x.." "..y)
  472.   local B = returnChar("Maze", x, y)
  473.   if B then
  474.     if B ~= "-" and B ~= "+" and B ~= "|" then
  475.       return true
  476.     end
  477.   end
  478.   return false
  479. end
  480.  
  481. while true do
  482.   event,param1 = os.pullEvent()
  483.   if event == "char" then
  484.     if param1 == "w" then
  485.       if safe(x, y-1) then
  486.         y = y-1
  487.         Char = "^"
  488.       end
  489.     elseif param1 == "s" then
  490.       if safe(x, y+1) then
  491.         y = y+1
  492.         Char = "v"
  493.       end
  494.     elseif param1 == "a" then
  495.       if safe(x-1, y) then
  496.         x = x-1
  497.         Char = "<"
  498.       end
  499.     elseif param1 == "d" then
  500.       if safe(x+1, y) then
  501.         x = x+1
  502.         Char = ">"
  503.       end
  504.     end
  505.     moveScreen("Main", x-xx, y-yy)
  506.     draw("Main")
  507.     term.setCursorPos(1,1)
  508.     local G = {returnView("Main")}
  509.     for n=1,#G do
  510.       term.setCursorPos(1,n)
  511.       term.write(G[n])
  512.     end
  513.     term.setCursorPos(xx, yy+1)
  514.     term.write(Char)
  515.   end
  516. end
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement