成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

canvas實(shí)例 --- 制作簡(jiǎn)易迷宮

twohappy / 843人閱讀

摘要:整個(gè)思路十分簡(jiǎn)單首先我們將迷宮視為一個(gè)行列的單元格組合,每一個(gè)單元格便可以表示為。用來(lái)儲(chǔ)存我們已訪問(wèn)過(guò)的單元格,則記錄我們的訪問(wèn)路徑。我們通過(guò)將單元格的,,,屬性設(shè)置為或來(lái)標(biāo)識(shí)這個(gè)方向是否應(yīng)該有邊框,同時(shí)該方向是否可走。

這個(gè)系列分為兩部分,第一部分為迷宮的生成及操作,第二部分為自動(dòng)尋路算法。

我們先看效果:點(diǎn)擊查看

我們直入正題,先說(shuō)一說(shuō)生成迷宮的思路。
整個(gè)思路十分簡(jiǎn)單:

首先我們將迷宮視為一個(gè)m行n列的單元格組合,每一個(gè)單元格便可以表示為 maze[i][j] 。接下來(lái)迷宮與m*n單元格的區(qū)別是什么呢?對(duì),迷宮就是相當(dāng)于不同單元格以某種規(guī)律相互連通,也就相當(dāng)于我們把相鄰的兩個(gè)單元格之間的重合線給去掉,然后按照某種規(guī)律循環(huán),便可生成一個(gè)迷宮。

我們假定從左上角開始出發(fā),遍歷每一個(gè)單元格,如果該單元格未被訪問(wèn)過(guò),則查看其相鄰元素(上,下,左,右)是否有未訪問(wèn)的單元格,如果有則隨機(jī)取出一個(gè)相鄰元素并打通他們之間的重合線,如果沒(méi)有則回退到上一個(gè)單元格。

上代碼:
首先我們創(chuàng)建一個(gè)構(gòu)造函數(shù):

function Maze(obj,col,row){
     this.col = col || 10;
     this.row = row || 10;
     this.canvas = obj.getContext("2d");
     this.init();
}

在這個(gè)構(gòu)造函數(shù)中,我們接收三個(gè)參數(shù),分別為canvas元素,迷宮的行數(shù)與列數(shù),并直接調(diào)用Maze的init方法。

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方法中,我們首先根據(jù)傳入的列數(shù)col來(lái)計(jì)算單元格的寬度,然后構(gòu)建一個(gè)maze_cells對(duì)象,其中每一行為一個(gè)數(shù)組,每個(gè)單元格包含的值分別代表x,y坐標(biāo),上下左右4個(gè)方向是否可以通行,是否訪問(wèn)過(guò),還有該單元格的g,h,f值。我們假定迷宮的開口位于整個(gè)迷宮的左上角,出口位于右下角。visitRooms用來(lái)儲(chǔ)存我們已訪問(wèn)過(guò)的單元格,roomLine則記錄我們的訪問(wèn)路徑。我們將迷宮的入口處和出口處的top,bottom分別設(shè)為true后再設(shè)置為false是為了在繪制的過(guò)程中不出現(xiàn)邊框,繪制完成后保證不能向上(下)移動(dòng)。

ps:canvas繪制線條是居中于我們坐標(biāo)的,即在(1,1)處繪制寬度為2的線條起始是從(0,1)開始的,所以我們用整個(gè)canvas的寬度減去了線條的寬度2,當(dāng)然這里也可以設(shè)置為變量更方便修改。

接下來(lái)我們需要遍歷每一個(gè)單元格,如下通過(guò)遞歸的形式訪問(wèn)每一個(gè)單元格,當(dāng)某一個(gè)單元格的相鄰元素全部被訪問(wèn)過(guò)并且roomLine數(shù)組為空時(shí)就意味著我們已經(jīng)訪問(wèn)了所有的單元格,具體原因自行腦補(bǔ)。

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 0){ //相鄰房間有未訪問(wèn)房間
        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)
        }
    }
},

我們看到如果當(dāng)前單元格的相鄰單元格有未訪問(wèn)的,則執(zhí)行breakWall方法,即打通當(dāng)前單元格與相鄰單元格中間的墻,當(dāng)然我們應(yīng)該隨機(jī)選擇一個(gè)未訪問(wèn)的相鄰單元格。我們通過(guò)將單元格的top,bottom,left,right屬性設(shè)置為true或false來(lái)標(biāo)識(shí)這個(gè)方向是否應(yīng)該有邊框,同時(shí)該方向是否可走。

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;
    }
},

進(jìn)行完上面的兩步,我們的一個(gè)完整數(shù)組已經(jīng)構(gòu)成了,接下來(lái)便可以開始繪制了,top,left,right,bottom為false時(shí)則有邊框,true時(shí)無(wú)邊框。這一步比較簡(jiǎn)單,我們?cè)诮Y(jié)尾調(diào)用了一個(gè)drawOffset方法,該方法將創(chuàng)建一個(gè)離屏對(duì)象,這樣我們?cè)趧?dòng)態(tài)修改迷宮的時(shí)候可以直接將離屏的圖像繪制到當(dāng)前畫布中。

drawCells : function(){
    var ctx = this.canvas,   //canvas對(duì)象
        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"
},

綁定事件比較簡(jiǎn)單,我們?yōu)閣indow監(jiān)聽keydown事件,根據(jù)不同的keyCode來(lái)判斷我們應(yīng)該行走的方向。

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("到達(dá)終點(diǎn)了")
    }
});

drawRect便是我們移動(dòng)的目標(biāo)。

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();
},

到這里我們的迷宮便完成了。

作者:易企秀——樊一

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/105979.html

相關(guān)文章

  • canvas實(shí)例 --- 制作簡(jiǎn)易迷宮

    摘要:整個(gè)思路十分簡(jiǎn)單首先我們將迷宮視為一個(gè)行列的單元格組合,每一個(gè)單元格便可以表示為。用來(lái)儲(chǔ)存我們已訪問(wèn)過(guò)的單元格,則記錄我們的訪問(wèn)路徑。我們通過(guò)將單元格的,,,屬性設(shè)置為或來(lái)標(biāo)識(shí)這個(gè)方向是否應(yīng)該有邊框,同時(shí)該方向是否可走。 這個(gè)系列分為兩部分,第一部分為迷宮的生成及操作,第二部分為自動(dòng)尋路算法。 我們先看效果:點(diǎn)擊查看 我們直入正題,先說(shuō)一說(shuō)生成迷宮的思路。 整個(gè)思路十分簡(jiǎn)單: 首先...

    孫吉亮 評(píng)論0 收藏0
  • SegmentFault 技術(shù)周刊 Vol.35 - WebGL:打開網(wǎng)頁(yè)看大片

    摘要:在文末,我會(huì)附上一個(gè)可加載的模型方便學(xué)習(xí)中文藝術(shù)字渲染用原生可以很容易地繪制文字,但是原生提供的文字效果美化功能十分有限。 showImg(https://segmentfault.com/img/bVWYnb?w=900&h=385); WebGL 可以說(shuō)是 HTML5 技術(shù)生態(tài)鏈中最為令人振奮的標(biāo)準(zhǔn)之一,它把 Web 帶入了 3D 的時(shí)代。 初識(shí) WebGL 先通過(guò)幾個(gè)使用 Web...

    objc94 評(píng)論0 收藏0
  • 快速實(shí)現(xiàn)一個(gè)簡(jiǎn)單的canvas迷宮游戲

    摘要:所以我先寫了一個(gè)樣例扔在服務(wù)器上,大家可以先體驗(yàn)一下效果用成就感作為驅(qū)動(dòng)力哈哈哈點(diǎn)我體驗(yàn)地址正文實(shí)現(xiàn)這個(gè)小游戲也不難,讓我們想想,一個(gè)迷宮游戲有哪些基本要素。迷宮地圖的生成,可以借助谷歌的一個(gè)迷宮在線生成器來(lái)獲得。 前言 (最近設(shè)計(jì)模式看的有點(diǎn)頭大,一直面對(duì)純js實(shí)在是有些枯燥-_-。所以寫一點(diǎn)有趣的東西調(diào)劑一下)現(xiàn)在canvas已經(jīng)不算新鮮了,不過(guò)由于日常業(yè)務(wù)中并不常用,所以實(shí)踐并不...

    wangdai 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<