Displaying a 3D Maze
Learn to display a 3D maze on the console in Ruby.
We'll cover the following...
Introduction
We’ll draw these mazes as sets of floor plans, with lower floors to the left and higher floors to the right. The up and down passages, represented by the red arrows in the corresponding cells, indicate which of the adjacent floors each passage leads to. For example, a 3x3x3 maze might look something like this:
Arrows pointing to the right are like stairs leading to the level above them, and arrows pointing to the left are stairs leading down. If we enter the maze in the northwest corner of the bottom level (the one on the far left), we might take two steps east to the northeast corner, take the stairs up to the same corner of the middle level, go one step to the west and then up another flight of stairs, winding up on the third level. It works, and it’s not too difficult to implement. To implement this, we have to update the Grid3D
class.
The updated Grid3D
class
require 'grid'class Cell3D < Cellattr_reader :levelattr_accessor :up, :downdef initialize(level, row, column)@level = levelsuper(row, column)enddef neighborslist = superlist << up if uplist << down if downlistendendclass Grid3D < Gridattr_reader :levelsdef initialize(levels, rows, columns)@levels = levelssuper(rows, columns)enddef prepare_gridArray.new(levels) do |level|Array.new(rows) do |row|Array.new(columns) do |column|Cell3D.new(level, row, column)endendendenddef configure_cellseach_cell do |cell|level, row, col = cell.level, cell.row, cell.columncell.north = self[level, row - 1, col]cell.south = self[level, row + 1, col]cell.west = self[level, row, col - 1]cell.east = self[level, row, col + 1]cell.down = self[level - 1, row, col]cell.up = self[level + 1, row, col]endenddef [](level, row, column)return nil unless level.between?(0, @levels - 1)return nil unless row.between?(0, @grid[level].count - 1)return nil unless column.between?(0, @grid[level][row].count - 1)@grid[level][row][column]enddef random_celllevel = rand(@levels)row = rand(@grid[level].count)column = rand(@grid[level][row].count)@grid[level][row][column]enddef size@levels * @rows * @columnsenddef each_level@grid.each do |level|yield levelendenddef each_roweach_level do |rows|rows.each do |row|yield rowendendenddef to_png(cell_size: 10, inset: 0, margin: cell_size/2)inset = (cell_size * inset).to_igrid_width = cell_size * columnsgrid_height = cell_size * rowsimg_width = grid_width * levels + (levels - 1) * marginimg_height = grid_heightbackground = ChunkyPNG::Color::WHITEwall = ChunkyPNG::Color::BLACKarrow = ChunkyPNG::Color.rgb(255, 0, 0)img = ChunkyPNG::Image.new(img_width + 1, img_height + 1, background)[:backgrounds, :walls].each do |mode|each_cell do |cell|x = cell.level * (grid_width + margin) + cell.column * cell_sizey = cell.row * cell_sizeif inset > 0to_png_with_inset(img, cell, mode, cell_size, wall, x, y, inset)elseto_png_without_inset(img, cell, mode, cell_size, wall, x, y)endif mode == :wallsmid_x = x + cell_size / 2mid_y = y + cell_size / 2if cell.linked?(cell.down)img.line(mid_x-3, mid_y, mid_x-1, mid_y+2, arrow)img.line(mid_x-3, mid_y, mid_x-1, mid_y-2, arrow)endif cell.linked?(cell.up)img.line(mid_x+3, mid_y, mid_x+1, mid_y+2, arrow)img.line(mid_x+3, mid_y, mid_x+1, mid_y-2, arrow)endendendendimgendend
Code explanation
Line 84: This implementation introduces a new parameter, ...