JS+Canvas實現(xiàn)貪吃蛇小游戲
今天呢,主要和小伙伴們分享一下一個貪吃蛇游戲從構(gòu)思到實現(xiàn)的過程~因為我不是很喜歡直接PO代碼,所以只copy代碼的童鞋們請出門左轉(zhuǎn)不謝。
按理說canvas與其應(yīng)用是老生常談了,可我在準(zhǔn)備階段卻搜索不到有用的資料(不是代碼?。哉f呢,只能自力更生 -_-
首先是大致要考慮的東西:
1.要有蛇(沒蛇怎么叫貪吃蛇)。
2.然后要有地圖(蛇是不能上天的)。
3.不能水平\垂直掉頭(如果想掉頭,需要至少變換方位并且至少移動一格才可)。
4.食物(不然怎么貪吃)。
5.吃了食物要變長(這才是精髓)。
PS:~現(xiàn)在我回想起來,當(dāng)時的確只想到這么多(⊙﹏⊙)
構(gòu)思完畢,開工!
怎么做呢?從大到小,先畫個矩形作地圖,可我覺得太丑,于是畫了一張圖出來:
context.beginPath(); var bgImg = new Image(); bgImg.src = "img/background.png"; context.drawImage(bgImg, 0, 0, 600, 600); context.closePath();
現(xiàn)在我們有地圖了

地圖上好像缺點什么……沒錯就是禮物,所以我們現(xiàn)在生成禮物,那么問題來了:禮物最多有幾個、生成位置、何時生成。
我這里暫時定義為:最多2個、隨機位置生成、當(dāng)禮物個數(shù)小于2時生成至2個。
接下來就很簡單了,上圖中,允許蛇活動的范圍是14顆樹(周圍兩顆樹是墻),然后16顆樹=600px,很容易我們得到每格多寬~
所以呢,我們只需要定義一個隨機生成1-14整數(shù)的方法就可以很輕松找到應(yīng)該生成的位置:
//隨機數(shù)
function selectfrom() {
return Math.floor(Math.random() * 14 + 1);
}然后再用求出的數(shù)乘以每一格子的寬度,即可求出生成的具體X坐標(biāo),因為是正方形,所以Y也一樣:
var x = selectfrom() * (600/16); var y = selectfrom() * (600/16);
并且每得到一組禮物坐標(biāo)后,都需要存儲在一個數(shù)組內(nèi)(一會兒有大用處),至于畫矩形太基礎(chǔ)我就不說了。
And Now,我們有了禮物,有了地圖,就差蛇了,那么問題又來了:出生的蛇多長、出生地、死亡方式、移動方式、轉(zhuǎn)彎方式、如何判斷吃掉了禮物、吃掉了禮物變長到哪里。
出生蛇長度:實際編寫過程中,我發(fā)現(xiàn)默認長度1和2都不能夠很好的體現(xiàn)“蛇的轉(zhuǎn)彎”,所以定義為3,并且需將蛇身所有坐標(biāo)記錄在數(shù)組內(nèi)。
出生地:地圖中央或者自己定一個位置(按照格子來分),XY坐標(biāo)求取方式上面已經(jīng)說過不再贅述。
死亡方式:碰到障礙,或者(吃到自己)蛇頭碰到蛇身。
移動方式:通過定義一個全局變量記錄當(dāng)前方向(0、1、2、3,默認1),并且使用計時器驅(qū)動蛇運動。
轉(zhuǎn)彎方式:加入鍵盤按鍵檢測事件,當(dāng)方向鍵按下的時候修改-記錄方向的全部變量即可。
如何判斷吃掉了禮物:每次蛇頭移動時,都要遍歷下禮物集合(上面有說過),如果蛇頭將要移動到的下個坐標(biāo)與之重合了,則視為吃掉了禮物。
吃掉了禮物變長到哪里:直接加在頭部可能會導(dǎo)致意外的死亡,所以我決定吃到禮物后的下一次移動不消除蛇尾(最后一個元素)。
有了上面的構(gòu)思,我們可以著手定義一些可能會用到的公共變量:
var canvas = document.getElementById("mycanvas");//畫布主體
var context = canvas.getContext("2d");
var timer;//計時器
const WIDTH = canvas.width;//畫布寬
const HEIGHT = canvas.height;//畫布高
const XSUM = 16; //畫布寬分為幾格
const YSUM = 15; //畫布高分為幾格
const MAXFFOD = 2; //最大食物數(shù)量
var score = 0;//定義記錄游戲得分
var xsplit = WIDTH / XSUM; //x每一格子的寬度
var ysplit = HEIGHT / YSUM; //y每一格子的高度
var foodcount = 0; //當(dāng)前食物數(shù)量
var sinak = []; //貪吃蛇坐標(biāo)集
var get = []; //禮物坐標(biāo)集
var MoveTo = 1; //移動方向 默認1(右)有了這些變量,是不是發(fā)現(xiàn)很多東西都通了呢?
我們先來畫蛇:
//畫貪吃蛇
function drawsinak(sl) { //sl默認長度
context.beginPath();
context.fillStyle = "#000";
var ling = 0; //貪吃蛇被打印長度
for (var r = 0; r < sinak.length; r++) {
context.fillRect(sinak[r].split(',')[0], sinak[r].split(',')[1], xsplit, ysplit);
ling++;
}
if (ling == 0) {
for (var i = 0; i < sl; i++) {
context.fillRect(xsplit * (7 - i), ysplit * 6, xsplit, ysplit); //默認出生點:7,6默認中心點
sinak.push(xsplit * (7 - i) + ',' + ysplit * 6);
}
}
context.fill();
context.closePath();
}可以看到我將生成的蛇的坐標(biāo)都計入了數(shù)組內(nèi),生成的禮物自然也要計入:
context.beginPath();
var x = selectfrom(XSUM - 2) * xsplit;
var y = selectfrom(YSUM - 2) * ysplit;
context.fillStyle = "red";
for (var i = 0; i < get.length; i++) {
context.fillRect(get[i].split(',')[0], get[i].split(',')[1], xsplit, ysplit);
context.fill();
foodcount++;
}
if (MAXFFOD > foodcount) {
context.fillRect(x, y, xsplit, ysplit);
context.fill();
foodcount++;
get.push(x + ',' + y);
}
context.closePath();接下來比較重要了,蛇的移動,以及吃到禮物和觸發(fā)死亡判斷:
//移動方法
//[c]移動方向 上右下左 0123
function sinakMove(c) {
context.beginPath();
//默認右側(cè)為頭
var tou = sinak[0]; //頭
var weiba = sinak[sinak.length - 1]; //尾巴
var oldX = tou.split(',')[0]; //頭部舊X坐標(biāo)
var oldY = tou.split(',')[1]; //頭部舊Y坐標(biāo)
var newX = 0; //頭部最新X坐標(biāo)
var newY = 0; //頭部最新Y坐標(biāo)
//計算頭部最新XY坐標(biāo)
switch (c) {
case 0:
newX = oldX;
newY = oldY - ysplit;
break;
case 1:
newX = (oldX - 0) + xsplit;
newY = oldY;
break;
case 2:
newX = oldX;
newY = (oldY - 0) + ysplit;
break;
case 3:
newX = oldX - xsplit;
newY = oldY;
break;
}
var flag = 0; //有沒有吃到禮物 0沒有1有
//如果吃到了禮物,則不消減尾部最后元素
for (var i = 0; i < get.length; i++) {
if (newX == get[i].split(',')[0] && newY == get[i].split(',')[1]) {
sinak.unshift(newX + ',' + newY);
foodcount--; //禮物計數(shù)減少1個
get.splice(i, 1); //清空禮物
flag = 1;
}
}
//如果沒有吃到禮物,則判斷是否碰到障礙或吃到自己
if (flag == 0) {
for (var i = 0; i < sinak.length; i++) {
if (newX == sinak[i].split(',')[0] && newY == sinak[i].split(',')[1]) {
if (confirm('吃掉了自己,游戲失??!是否重新開始?')) {
location.reload(true);
} else {
context.clearRect(0, 0, WIDTH, HEIGHT);
}
}
}
if (xsplit * (XSUM - 2) < newX || ysplit * (YSUM - 2) < newY || newX == 0 || newY == 0) {
if (confirm('撞墻了,游戲失?。∈欠裰匦麻_始?')) {
location.reload(true);
}
}
}
//如果沒有吃到禮物,那么進行普通移動
if (flag == 0) {
sinak.unshift(newX + ',' + newY);
sinak.splice(sinak.length - 1, 1);
}
//畫蛇
for (var r = 0; r < sinak.length; r++) {
context.fillRect(sinak[r].split(',')[0], sinak[r].split(',')[1], xsplit, ysplit);
}
context.closePath();
}控制蛇的方向:
//鍵盤事件
document.onkeydown = function (event) {
var e = event || window.event || arguments.callee.caller.arguments[0];
var move = 0; //移動方向
if (e && e.keyCode == 37) { //左
move = (MoveTo == 1 ? 1 : 3);
} else if (e && e.keyCode == 38) { //上
move = (MoveTo == 2 ? 2 : 0);
} else if (e && e.keyCode == 39) { //右
move = (MoveTo == 3 ? 3 : 1);
} else if (e && e.keyCode == 40) { //下
move = (MoveTo == 0 ? 0 : 2);
} else if (e && e.keyCode == 32) {//暫停游戲
clearInterval(timer);
}
MoveTo = move; //修改當(dāng)前移動方向
};這里做了防誤操作,當(dāng)蛇正在朝向某方向移動時,直接輸入反方向是無效的。如:蛇正向右走,這時直接按←鍵是無效的,仍然往右走。
一路跟著做到這里,相信大家的貪吃蛇已經(jīng)可以正常游戲了,不過我這個做的很糙,大家可以加入一些自己的想法,比如:
計分通關(guān),通關(guān)之后通過加快蛇的移動速度來增加難度。
隨機生成多種果實,比如加速果實,雙倍成長果實等。
加入WebSocket,實現(xiàn)網(wǎng)絡(luò)版貪吃蛇。
我寫過關(guān)于WebSocket的實現(xiàn),有興趣的也可以去看看,下面是鏈接:
基于SuperSocket實現(xiàn)的WebSocket(后端)
基于SuperSocket實現(xiàn)的WebSocket(前端)
到此這篇關(guān)于JS+Canvas實現(xiàn)貪吃蛇小游戲的文章就介紹到這了。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
javascript 3d 逐偵產(chǎn)品展示(核心精簡)
這篇文章主要介紹了javascript實現(xiàn)的3d逐偵產(chǎn)品展示,需要的朋友可以參考下2014-03-03
javascript實現(xiàn)當(dāng)前頁導(dǎo)航激活的方法
這篇文章主要介紹了javascript實現(xiàn)當(dāng)前頁導(dǎo)航激活的方法,涉及javascript操作css的技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-02-02
JavaScript中ES6規(guī)范中l(wèi)et和const的用法和區(qū)別
這篇文章主要介紹了JavaScript中ES6規(guī)范中l(wèi)et和const的用法和區(qū)別,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
JavaScript數(shù)組Array對象增加和刪除元素方法總結(jié)
這篇文章主要介紹了JavaScript數(shù)組Array對象增加和刪除元素方法,實例總結(jié)了pop方法、push方法、splice方法、concat方法等的使用技巧,需要的朋友可以參考下2015-01-01

