這個系列分爲兩部分,第一部分爲迷宮的生成及操做,第二部分爲自動尋路算法。算法
咱們先看效果:canvas
See the Pen QGKBjm by fanyipin (@fanyipin) on CodePen.數組
咱們直入正題,先說一說生成迷宮的思路。app
整個思路十分簡單:dom
首先咱們將迷宮視爲一個m行n列的單元格組合,每個單元格即可以表示爲maze[i][j]。接下來迷宮與m*n單元格的區別是什麼呢?對,迷宮就是至關於不一樣單元格以某種規律相互連通,也就至關於咱們把相鄰的兩個單元格之間的重合線給去掉,而後按照某種規律循環,即可生成一個迷宮。函數
咱們假定從左上角開始出發,遍歷每個單元格,若是該單元格未被訪問過,則查看其相鄰元素(上,下,左,右)是否有未訪問的單元格,若是有則隨機取出一個相鄰元素並打通他們之間的重合線,若是沒有則回退到上一個單元格。this
上代碼:spa
首先咱們建立一個構造函數:rest
function Maze(obj,col,row){ this.col = col || 10; this.row = row || 10; this.canvas = obj.getContext('2d'); this.init(); }
在這個構造函數中,咱們接收三個參數,分別爲canvas元素,迷宮的行數與列數,並直接調用Maze的init方法。code
init : function(){ this.cell = (width - 2) / this.col; for(var i = 0 ; i < this.row ; i++){ maze_cells[i] = []; for(var j = 0; j < this.col ; j++){ maze_cells[i].push({ 'x' : j, 'y' : i, 'top' : false, 'bottom' : false, 'left' : false, 'right' : false, 'isVisited' : false, 'g' : 0, 'h' : 0, 'f' : 0 }) } } start_cell = {'x' : 0, 'y' : 0 }; start_row = start_cell.x; start_col = start_cell.y; visitRooms.push(start_cell) roomsLine.push(start_cell) maze_cells[0][0].isVisited = true; maze_cells[0][0].top = true; maze_cells[this.row-1][this.col-1].bottom = true; this.calcCells(0,0,maze_cells); this.drawCells(); maze_cells[0][0].top = false; maze_cells[this.row-1][this.col-1].bottom = false; this.drawRect(start_col,start_row); this.bindEvent(); },
在init方法中,咱們首先根據傳入的列數col來計算單元格的寬度,而後構建一個maze_cells對象,其中每一行爲一個數組,每一個單元格包含的值分別表明x,y座標,上下左右4個方向是否能夠通行,是否訪問過,還有該單元格的g,h,f值。咱們假定迷宮的開口位於整個迷宮的左上角,出口位於右下角。visitRooms用來儲存咱們已訪問過的單元格,roomLine則記錄咱們的訪問路徑。咱們將迷宮的入口處和出口處的top,bottom分別設爲true後再設置爲false是爲了在繪製的過程當中不出現邊框,繪製完成後保證不能向上(下)移動。
ps:canvas繪製線條是居中於咱們座標的,即在(1,1)處繪製寬度爲2的線條起始是從(0,1)開始的,因此咱們用整個canvas的寬度減去了線條的寬度2,固然這裏也能夠設置爲變量更方便修改。
接下來咱們須要遍歷每個單元格,以下經過遞歸的形式訪問每個單元格,當某一個單元格的相鄰元素所有被訪問過而且roomLine數組爲空時就意味着咱們已經訪問了全部的單元格,具體緣由自行腦補。
calcCells : function(x,y,arr){ var neighbors = []; if(x-1 >=0 && !maze_cells[x-1][y].isVisited){ neighbors.push({'x' : x-1 ,'y' : y}) } if(x+1 < this.row && !maze_cells[x+1][y].isVisited){ neighbors.push({'x' : x+1 ,'y' : y}) } if(y-1 >=0 && !maze_cells[x][y-1].isVisited){ neighbors.push({'x' : x ,'y' : y-1}) } if(y+1 <this.col && !maze_cells[x][y+1].isVisited){ neighbors.push({'x' : x ,'y' : y+1}) } if(neighbors.length>0){ //相鄰房間有未訪問房間 var current = {'x' : x , 'y' : y}; var next = neighbors[Math.floor(Math.random() * neighbors.length)]; maze_cells[next.x][next.y].isVisited = true; visitRooms.push({'x' : next.x , 'y' : next.y}) roomsLine.push({'x' : next.x , 'y' : next.y}); this.breakWall(current,next); this.calcCells(next.x,next.y,arr) }else{ var next = roomsLine.pop(); if(next != null){ this.calcCells(next.x,next.y,arr) } } },
咱們看到若是當前單元格的相鄰單元格有未訪問的,則執行breakWall方法,即打通當前單元格與相鄰單元格中間的牆,固然咱們應該隨機選擇一個未訪問的相鄰單元格。咱們經過將單元格的top,bottom,left,right屬性設置爲true或false來標識這個方向是否應該有邊框,同時該方向是否可走。
breakWall : function(cur,next){ if(cur.x < next.x){ maze_cells[cur.x][cur.y].bottom = true; maze_cells[next.x][next.y].top = true; } if(cur.x > next.x){ maze_cells[cur.x][cur.y].top = true; maze_cells[next.x][next.y].bottom = true; } if(cur.y < next.y){ maze_cells[cur.x][cur.y].right = true; maze_cells[next.x][next.y].left = true; } if(cur.y > next.y){ maze_cells[cur.x][cur.y].left = true; maze_cells[next.x][next.y].right = true; } },
進行完上面的兩步,咱們的一個完整數組已經構成了,接下來即可以開始繪製了,top,left,right,bottom爲false時則有邊框,true時無邊框。這一步比較簡單,咱們在結尾調用了一個drawOffset方法,該方法將建立一個離屏對象,這樣咱們在動態修改迷宮的時候能夠直接將離屏的圖像繪製到當前畫布中。
drawCells : function(){ var ctx = this.canvas, //canvas對象 w = this.cell; ctx.clearRect(0,0,$('canvas').width,$('canvas').height) ctx.beginPath(); ctx.save(); ctx.translate(1,1) ctx.strokeStyle = '#000000'; ctx.lineWidth = 2; for(var i in maze_cells){ //i 爲 row var len = maze_cells[i].length; for( var j = 0; j < len; j++){ var cell = maze_cells[i][j]; i = parseInt(i); if(!cell.top){ ctx.moveTo(j*w,i*w); ctx.lineTo((j+1)*w ,i*w); } if(!cell.bottom){ ctx.moveTo(j*w,(i+1)*w); ctx.lineTo((j+1)*w ,(i+1)*w) } if(!cell.left){ ctx.moveTo(j*w,i*w); ctx.lineTo(j*w,(i+1)*w ) } if(!cell.right){ ctx.moveTo((j+1)*w,i*w); ctx.lineTo((j+1)*w,(i+1)*w) } } } ctx.stroke(); ctx.restore(); this.drawOffset(); },
drawOffset : function(){ var offsetCanvas = document.createElement('canvas'); offsetCanvas.id = 'offset'; document.body.appendChild(offsetCanvas); offsetCanvas.width = $('canvas').width; offsetCanvas.height = $('canvas').height; var offset = $('offset').getContext('2d'); offset.clearRect(0,0,$('canvas').width,$('canvas').height) offset.drawImage($('canvas'),0,0,offsetCanvas.width,offsetCanvas.height); $('offset').style.display ='none' },
綁定事件比較簡單,咱們爲window監聽keydown事件,根據不一樣的keyCode來判斷咱們應該行走的方向。
var _self = this; window.addEventListener('keydown',function(event){ switch (event.keyCode) { case 37 : event.preventDefault(); if(maze_cells[start_row][start_col].left){ start_col --; } break; case 38 : event.preventDefault(); if(maze_cells[start_row][start_col].top){ start_row --; } break; case 39 : event.preventDefault(); if(maze_cells[start_row][start_col].right){ start_col ++ } break; case 40 : event.preventDefault(); if(maze_cells[start_row][start_col].bottom){ start_row ++; } break; } _self.drawRect(start_col,start_row); if(start_col == (_self.col - 1) && start_row == ( _self.row - 1)){ alert('到達終點了') } });
drawRect即是咱們移動的目標。
drawRect : function(col,row){ var ctx = this.canvas; ctx.save(); ctx.clearRect(0,0,canvas.width,canvas.height); ctx.drawImage($('offset'),0,0) ctx.translate(2,2) ctx.fillStyle = '#ff0000'; ctx.fillRect(col*this.cell,row*this.cell,this.cell-2,this.cell-2); ctx.restore(); },
到這裏咱們的迷宮便完成了。