Vue实现开心消消乐游戏算法


Posted in Javascript onOctober 22, 2019

之前做过一个算法题,算法要求就是写一个开心消消乐的逻辑算法,当时也是考虑了一段时间才做出来。后来想了想,既然核心算法都有了,能不能实现一个开心消消乐的小游戏呢,于是花了两天时间做了一个小游戏出来。

效果展示#

先在这里放一个最终实现的效果,还是一个比较初级的版本,大家有什么想法欢迎评论哦

Vue实现开心消消乐游戏算法

游戏规则:

初始时会给玩家十分的初始分,每拖动一次就减一分,每消除一个方块就加一分,直到最后分数为0游戏结束

任意两个方块都可以拖动

界面设计#

页面的布局比较简单,格子的数据是一个二维数组的形式,说到这里大家应该已经明白界面是怎么做的了。

<div
 v-for="(item, index) in squareData"
 :key="index"
 class="row">
 <div
  v-for="(_item, _index) in item"
  :key="_index"
  class="square"
  :class="_item"
  @mousedown="dragStart(index, _index)"
  @mouseup="dragEnd">
  {{_item}}
 </div>
</div>

大家应该注意到了 :class="_item" 的写法,动态命名class,使得其每个种类的方块的颜色都不同,最后可以按照同色消除的玩法就行操作。

.square.A{
 background-color: #8D98CA;
}
.square.S{
 background-color: #A9A2F6;
}
/*其余操作相同*/

同时在玩家点击方块的时候方块会左右摆动以表示选中了此方块,还可以提升游戏的灵动性。关于HTML动画的实现方式有很多,在这里我们使用CSS animation进行操作,代码如下:

@keyframes jitter {
 from, 50%, to {
  transform: rotate(0deg);
 }
 10%, 30% {
  transform: rotate(10deg);
 }
 20% {
  transform: rotate(20deg);
 }
 60%, 80% {
  transform: rotate(-10deg);
 }
 70% {
  transform: rotate(-20deg);
 }
}
/* 只要是用户点击不动,动画就不会停止 */
.square:active{
 animation-name: jitter;
 animation-duration: 0.5s;
 animation-iteration-count: infinite;
}

核心算法#

消除算法

上面提到我之前是做过一道题是判断一个二维数组中有没有可消的元素,有的话是多少个。

在这里我们可以这样想,最开始遍历一整个二维数组,每次定义一个 X0 , X1 , Y0, Y1, 然后每次计算其上下左右连续相同方块的位置,在这个过程中要注意边界问题,然后我们记录下这四个变量,只要 |X0-X1+1|>=3 或者 |Y0-Y1+1|>=3,我们就可以将这个方块的坐标加入到 del数组中。

遍历完一整个二维数组之后,我们就可以将 del数组中对应坐标位置的方块内容变为 '0', 由于我们没有对 0 定义样式,所以在没有执行下落算法之前变为 0 的方块为白色。

下落算法

在我们将相应的方块白色之后,其上面的方块应该下落,在这里我的思想是这个样子的。

按照列遍历二维数组,定义一个指针 t,指向上次不为 0 的方块位置,一旦遇到方块不为 0 的格子就将其与t所指的方块就行交换,一次类推,示意图如下:

Vue实现开心消消乐游戏算法

这样的话我们就可以把为空的上移到最顶层,并且不打乱顺序,然后我们在随机填充顶部的空方块就可以了。做完填充之后我们要再做一次消除算法,直到del数组的长度为空为止,这个道理大家应该都能想得到。

代码如下

clear(): void {
 const m: number = 10;
 const n: number = 10;
 while (true) {
  const del: any[] = [];
  for (let i: number = 0; i < m; i++) {
   for (let j: number = 0; j < n; j++) {
    if (this.squareData[i][j] === '0') {
     continue;
    }
    let x0: number = i;
    let x1: number = i;
    let y0: number = j;
    let y1: number = j;
    while (x0 >= 0 && x0 > i - 3 && this.squareData[x0][j] === this.squareData[i][j]) {
     --x0;
    }
    while (x1 < m && x1 < i + 3 && this.squareData[x1][j] === this.squareData[i][j]) {
     ++x1;
    }
    while (y0 >= 0 && y0 > j - 3 && this.squareData[i][y0] === this.squareData[i][j]) {
     --y0;
    }
    while (y1 < n && y1 < j + 3 && this.squareData[i][y1] === this.squareData[i][j]) {
     ++y1;
    }
    if (x1 - x0 > 3 || y1 - y0 > 3) {
     del.push([i, j]);
    }
   }
  }
  if (del.length === 0) {
   break;
  }
  this.score += del.length;
  for (const square of del) {
   this.$set(this.squareData[square[0]], square[1], '0');
  }
  for (let j: number = 0; j < n; ++j) {
   let t: number = m - 1;
   for (let i: number = m - 1; i >= 0; --i) {
    if (this.squareData[i][j] !== '0') {
     [this.squareData[t][j], this.squareData[i][j]] = [this.squareData[i][j], this.squareData[t][j]];
     t -= 1;
    }
   }
  }
 }
},

游戏结束#

分数为 0 的时候游戏结束,此时在执行一遍初始化函数,重新生成一个开心消消乐格子,将分数初始化为10.

if (this.score <= 0) {
  if (confirm('分数用光了哦~~')) {
   this.init();
  } else {
   this.init();
  }
 }

项目源代码#

目前项目是在github上托管,欢迎PR!点此跳转

总结

以上所述是小编给大家介绍的Vue实现开心消消乐算法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

Javascript 相关文章推荐
Code:loadScript( )加载js的功能函数
Feb 02 Javascript
JS+DIV实现鼠标划过切换层效果的实例代码
Nov 26 Javascript
JavaScript获得指定对象大小的方法
Jul 01 Javascript
使用three.js 画渐变的直线
Jun 05 Javascript
JavaScript sort数组排序方法和自我实现排序方法小结
Jun 06 Javascript
JS创建对象的写法示例
Nov 04 Javascript
vue.js加载新的内容(实例代码)
Jun 01 Javascript
详解Vue + Vuex 如何使用 vm.$nextTick
Nov 20 Javascript
小程序实现带年月选取效果的日历
Jun 27 Javascript
JS 正则表达式验证密码、邮箱格式的实例代码
Oct 28 Javascript
mpvue开发音频类小程序踩坑和建议详解
Mar 12 Javascript
vue组件的路由高亮问题解决方法
May 11 Vue.js
p5.js码绘“跳动的小正方形”的实现代码
Oct 22 #Javascript
微信小程序绑定手机号获取验证码功能
Oct 22 #Javascript
微信小程序开发之转发分享功能
Oct 22 #Javascript
微信小程序转发事件实现解析
Oct 22 #Javascript
小程序点击图片实现png转jpg
Oct 22 #Javascript
Node.js 实现抢票小工具 &amp; 短信通知提醒功能
Oct 22 #Javascript
微信小程序实现左侧滑动导航栏
Apr 08 #Javascript
You might like
php不使用插件导出excel的简单方法
2014/03/04 PHP
PHP中文字符串截断无乱码解决方法
2016/10/10 PHP
jquery获取对象的方法足以应付常见的各种类型的对象
2014/05/14 Javascript
IE下使用jQuery重置iframe地址时内存泄露问题解决办法
2015/02/05 Javascript
基于jQuery实现拖拽图标到回收站并删除功能
2015/11/25 Javascript
一系列Bootstrap导航条使用方法分享
2016/04/29 Javascript
使用Web Uploader实现多文件上传
2016/06/08 Javascript
全面解析Bootstrap中tooltip、popover的使用方法
2016/06/13 Javascript
同步异步动态引入js文件的几种方法总结
2016/09/23 Javascript
JavaScrip数组去重操作实例小结
2019/06/20 Javascript
javascript实现自由编辑图片代码详解
2019/06/21 Javascript
JS实现电商商品展示放大镜特效
2020/01/07 Javascript
Python实现的石头剪子布代码分享
2014/08/22 Python
跟老齐学Python之dict()的操作方法
2014/09/24 Python
python使用marshal模块序列化实例
2014/09/25 Python
在Python中操作字典之clear()方法的使用
2015/05/21 Python
Python内建数据结构详解
2016/02/03 Python
Python之Scrapy爬虫框架安装及简单使用详解
2017/12/22 Python
python 实现任务管理清单案例
2020/04/25 Python
Python中无限循环需要什么条件
2020/05/27 Python
使用Python文件读写,自定义分隔符(custom delimiter)
2020/07/05 Python
匡威帆布鞋美国官网:Converse美国
2016/08/22 全球购物
Sephora丝芙兰澳洲官方网站:国际知名化妆品购物
2016/10/27 全球购物
JBL美国官方商店:扬声器、耳机等
2019/12/01 全球购物
简短的公司员工自我评价分享
2013/11/13 职场文书
职业生涯规划设计步骤
2014/01/12 职场文书
银行存款证明样本
2014/01/17 职场文书
《兰亭集序》教学反思
2014/02/11 职场文书
校庆活动方案
2014/03/31 职场文书
领导班子个人对照检查材料(群众路线)
2014/09/26 职场文书
四风对照检查材料范文
2014/09/27 职场文书
教师个人师德总结
2015/02/06 职场文书
大学生年度个人总结
2015/02/15 职场文书
2015年保险公司工作总结
2015/04/24 职场文书
一封真诚的自荐信帮你赢得机会
2019/05/07 职场文书
css样式important规则的正确使用方式
2022/06/10 HTML / CSS