扫雷

在以前的win7系统中,自带的小游戏中,有一款游戏叫扫雷,效果如下图:

扫雷

html中的结构代码如下:

<select name="mineNum">
    <option value="0">请选择雷的数量</option>
    <option value="10-8">10个雷</option>
    <option value="40-16">40个雷</option>
    <option value="99-32">99个雷</option>
</select>
剩余雷数量:<span class="mineNum">0</span>

js逻辑代码中,预置两个工具函数:

// 设置样式的函数
function setStyle(ele, styleObj){
    for(var attr in styleObj){
        ele.style[attr] = styleObj[attr];
    }
}
// 获取随机数的函数
function getRandom(a,b=0){
    var max = a;
    var min = b;
    if(a<b){
        max = b;
        min = a;
    }
    return Math.floor(Math.random() * (max - min)) + min;
}

首先要根据选中的雷的数量,来创建小盒子:

// 获取标签
var mineSelect = document.querySelector('[name="mineNum"]');
var mineNumBox = document.querySelector('.mineNum');
// 获取扫雷游戏需要的列的数量
mineSelect.onchange = function(){
    // 获取当前选中的option的value
    var value = this.value;
    if(value === '0'){
        alert("请选择雷的数量")
        return false;
    }
    // 从中获取到类的数量和扫雷需要的列的数量
    var mineNum = +value.split('-')[0]
    var col = +value.split('-')[1]
    // 如果页面中已经有扫雷游戏了,就将之前的删除
    var mine = document.querySelector('.mine');
    if(mine){
        document.body.removeChild(mine)
    }
    // 调用创建扫雷游戏的函数
    createMine(mineNum,col)
}
// 创建扫雷游戏的函数
function createMine(mineNum,col){
    // 根据列的数量创建大盒子
    var box = document.createElement('div')
    // 设置类名
    box.className = 'mine';
    document.body.appendChild(box)
    // 不能选中box中的内容
    box.onselectstart = function(){
        return false
    }
    // 设置样式
    setStyle(box,{
        border:"3px solid #00f",
        position:"relative",
        width:col*20 + 'px',
        height:col*20 + 'px',
    })
    // 在大盒子中创建小盒子
    for(var i=0;i<Math.pow(col,2);i++){
        var div = document.createElement('div')
        box.appendChild(div)
        setStyle(div,{
            width:"18px",
            height:"18px",
            border:"1px solid #fff",
            position:"absolute",
            left:i%col*20+'px',
            top:Math.floor(i/col)*20 + 'px',
            backgroundColor:"#aaa"
        })
    }
    // 调用随机设置雷的函数
    var indexArr = setMine(box,mineNum,col)
    // 调用计算雷数量的函数
    countMine(box,col)
    // 点击小div开始扫
    clearance(box,col,indexArr)
}

设置雷的函数如下:

// 设置随机雷的函数
function setMine(ele,num,col){
    // 在ele中创建num个雷
    // 定义数组用来存放是雷的小div的下标
    var indexArr = []
    for(var i=0;i<num;i++){
        // 获取随机下标
        var randomIndex = getRandom(ele.children.length)
        // 判断随机下标是否在数组中
        var index = indexArr.indexOf(randomIndex)
        // 如果这个随机下标不在数组中,就将这个随机下标放当数组中
        if(index<0){
            indexArr.push(randomIndex)
        }else{
            // 这次循环作废 - 重新创建随机下标 - 要保证数组中随机下标的个数一定是雷的数量
            i--
        }
    }
    // indexArr中存放的是所有 是雷的div的下标
    // 给所有是雷的div做特殊的标记
    for(var i=0;i<indexArr.length;i++){
        ele.children[indexArr[i]].mine = true;
        // ele.children[indexArr[i]].style.backgroundColor = 'red';
    }
    // 更改页面中雷的数量
    mineNumBox.innerText = num
    return indexArr
}

根据设置好的雷,给每个小div计算周围雷的数量:

// 计算雷数量的函数
function countMine(ele,col){
    // 遍历每个小div,计算周围雷的数量
    for(var i=0;i<ele.children.length;i++){
        // 如果当前div是雷就跳过
        if(ele.children[i].mine){
            continue
        }
        // 获取周围所有div的下标的数组
        var arr = getIndexArr(i,col);
        // 定义当前div周围雷的数量变量
        var num = 0
        // 遍历周围的div计算
        for(var j=0;j<arr.length;j++){
            if(ele.children[i+arr[j]].mine){
                num++
            }
        }
        ele.children[i].num = num;
        // ele.children[i].innerText = num;
    }
}

其中获取周围div的下标数组的代码如下:

// 获取每个div周围的div下标的数组
function getIndexArr(index,col){
    // 定义周围的div下标的数组
    var arr = [1,-1,col,-col,col+1,col-1,-col-1,-col+1];
    if(index<col){
        arr = [-1,1,col,col-1,col+1];
    }
    // 如果是最后1行,周围只有5个div - arr数组中就应该只有5个下标
    if(Math.floor(index/col) === col-1){
        arr = [-1,1,-col,-col-1,-col+1];
    }
    // 如果是第1列,周围只有5个div - arr数组中就应该只有5个下标
    if(index%col === 0){
        arr = [col,col+1,1,-col,-col+1];
    }
    // 如果是最后1列,周围只有5个div - arr数组中就应该只有5个下标
    if((index+1)%col === 0){
        arr = [-col-1,-col,-1,col-1,col];
    }
    // 如果是左上角的div,周围只有3个div - arr数组中就应该只有3个下标
    if(index===0){
        arr = [1,col,col+1];
    }
    // 如果是左下角的div,周围只有3个div - arr数组中就应该只有3个下标
    if(index===(col-1)*col){
        arr = [1,-col,-col+1];
    }
    // 如果是右下角的div,周围只有3个div - arr数组中就应该只有3个下标
    if(index===col*col-1){
        arr = [-1,-col,-col-1];
    }
    // 如果是右上角的div,周围只有3个div - arr数组中就应该只有3个下标
    if(index===col-1){
        arr = [-1,col,col-1];
    }
    return arr
}

再下来就可以开始玩游戏了,开始扫雷的代码如下:

// 点击小div开始扫的函数
function clearance(ele,col,indexArr){
    // 遍历所有小div绑定事件
    for(let i=0;i<ele.children.length;i++){
        // 单击事件 - 如果不是雷,就将周围雷的数量显示出来 - 如果周围没有雷就不显示数量,继续将周围的div点开
        ele.children[i].onclick = function(){
            if(!this.mine){
                // 如果点击的不是雷,就打开
                openNow(this,i,col,ele)
            }else{
                alert("GAME OVER");
                // 如果点击的div是雷,就将所有类引爆,并结束游戏
                for(var j=0;j<indexArr.length;j++){
                    ele.children[indexArr[j]].style.backgroundColor = 'red';
                }
            }
        }
        // 右击事件,标记雷
        ele.children[i].oncontextmenu = function(){
            // 将雷标红
            this.style.backgroundColor = 'red';
            mineNumBox.innerText = mineNumBox.innerText-1
            // 阻止默认行为
            return false;
        }
    }
}

打开当前div的函数如下:

// 打开当前小div,设置不同的背景颜色,并判断是否需要递归打开
function openNow(nowEle,i,col,ele){
    // 给已经打开的div做标记
    nowEle.open = true;
    // 设置打开的div的背景颜色
    nowEle.style.backgroundColor = '#eee';
    // 如果数量是0就继续打开
    if(nowEle.num === 0){
        // 继续打开周围的div
        open(ele,i,col)
    }else{
        // 如果数量不是0就显示数量
        nowEle.innerText = nowEle.num;
        nowEle.style.textAlign = 'center'
        nowEle.style.lineHeight = '20px'
        nowEle.style.fontSize = '12px'
        nowEle.style.color = '#666'
    }
}

如果当前div周围雷的数量为0,就将周围的div也打开,需要递归,函数如下:

// 继续打开周围div的函数
function open(ele,index,col){
    // 遍历周围的div - 如果雷的数量不是0,就显示数量,如果雷的数量是雷就继续递归打开
    var arr = getIndexArr(index,col);
    // 遍历数组,判断周围div是否打开
    for(var i=0;i<arr.length;i++){
        if(ele.children[index+arr[i]].open){
            continue;
        }else{
            openNow(ele.children[index+arr[i]],index+arr[i],col,ele)
        }
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,711评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,079评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,194评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,089评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,197评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,306评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,338评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,119评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,541评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,846评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,014评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,694评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,322评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,026评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,257评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,863评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,895评论 2 351

推荐阅读更多精彩内容

  • 我在编程教学方面不是专家,但当我想更好掌握某一样东西时,会试着找出让自己乐在其中的方法。比方说,当我想在shell...
    老率的IT私房菜阅读 124评论 0 0
  • 前篇: web版扫雷开发小记(1)web版扫雷开发小记(2)web版扫雷开发小记(3)web版扫雷开发小记(4) ...
    franose阅读 564评论 0 0
  • 一、 扫雷游戏实现核心思路解析 数据和视图尽量分离。采用面向对象的实现设计数据模块。格子作为一类对象,雷场作为一类...
    游戏开发大表哥阅读 518评论 0 0
  • 16宿命:用概率思维提高你的胜算 以前的我是风险厌恶者,不喜欢去冒险,但是人生放弃了冒险,也就放弃了无数的可能。 ...
    yichen大刀阅读 6,042评论 0 4
  • 公元:2019年11月28日19时42分农历:二零一九年 十一月 初三日 戌时干支:己亥乙亥己巳甲戌当月节气:立冬...
    石放阅读 6,877评论 0 2