大三学生的游戏开发之旅-Cocos Creator微信小游戏实战(球球合成-2048)项目开发思路解析/方案

本文主要是介绍大三学生的游戏开发之旅-Cocos Creator微信小游戏实战(球球合成-2048)项目开发思路解析/方案,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在上一篇文章中提到参加了一个小游戏竞赛,这篇文章就来讲解这个项目的基本思路(主要提供开发思路及部分代码),


这个项目主要就是一个2048的改版,玩家使用小飞船来左右移动控制生成小球,并在玩家手指离开屏幕时小球掉落,与场上其他小球进行合成(当小球合成到2048时,消除小球),目前主要是合成玩法(有个在职策划小哥哥推荐把这个项目做成解压+关卡玩法,但由于时间关系,最终没有实现,有兴趣的可以自己试试)。下面为这个项目的小程序码,感兴趣的可以去试一试(体验一下),希望为更多开发者提供良好思路。

        


1:微信方面

在和微信开发者工具搭配使用方面主要使用微信的云开发和微信授权登录,用户分享转发等功能,其中,创建全屏登录按钮的部分代码如下。

                let width = window.wx.getSystemInfoSync().screenWidth;//获取设备宽屏幕let height = window.wx.getSystemInfoSync().screenHeight;//获取设备屏幕高var button = wx.createUserInfoButton({type: 'text',text: '',style: {left:0,//距左0top:0,//距顶0width:width,//按钮宽度height:height,//按钮高度}});button.show();

2:新手教程

新手教程主要用到了一个本地缓存的功能,玩家第一次进入游戏时,判断一个本地缓存数据是否存在(由于玩家首次进入,肯定是不存在的啦),不存在则新手教程开启,当玩家做出了指定操作的时候(玩家手指离开屏幕成功生成一个小球后),往本地缓存写入一个数据,那么,当下次玩家游戏的时候会进行该本地缓存数据判断,这个时候,本地存在该数据,则新手教程关闭,新手教程的节点随之销毁。

2.1:判断是否存在本地缓存数据

maskBtn() {var playerP = cc.sys.localStorage.getItem('oldPlayer');//拿出本地缓存的数据if(playerP==1){//如果本地缓存的数据存在且为指定数据this.hand.destroy()//新手教程节点销毁}else{this.hand.active=true;//新手教程节点开启(刚开始进入为关闭)}},

2.2:若不存在本地缓存数据,则在touchEnd后,往本地写入数据

onTouchEnd() {this.hand.destroy();//新手指引销毁if(!playerP){//如果是新用户var p = 1;cc.sys.localStorage.setItem('oldPlayer',p);//写入数据};var playerP = cc.sys.localStorage.getItem('oldPlayer');console.log(playerP);//打印一下看是否正确}},

3:飞船移动

在玩家控制飞船的左右移动事件中,主要是开始触摸屏幕,移动以及结束触摸操作,在开始移动时,飞船下生成小球,移动过程中获取触摸点的位置变化然后引起飞船的位置变化,触摸结束将小球重力缩放设置为1(小球会向下掉落),并将掉落的小球绑定到另一父节点上。在这些操作中,可能会存在飞船飞出屏幕左右边缘的情况,于是用一个scene()函数控制飞船处于屏幕内。

3.1:飞船移动控制

onTouchStart(e) {if (this.state == 1) {// console.log("start");//记录数值this.thisNum = this.nextNum.string;//初始化球this.initBallNode();// 球赋值this.ballNode.getComponent('ballNode').updatestr(this.thisNum)//positionthis.ballNode.setPosition(e.getDelta());//-44this.ballNode.y = -44}},
onTouchEnd() {if (this.state == 1) {//开启重力this.ballNode.getComponent('ballNode').initGra(2)// console.log("end");/**数值更新 */this.updateNum();//换绑父节点this.ballNodeFall();//获取最大的数值for (var a = 0; a < gNumArr.length; a++) {this.arr = Math.max.apply(null, gNumArr)}// console.log(gNumArr)// console.log(this.arr)if (this.arr == 8) { x = 1 }//  一个小球生成机制if (this.arr == 16) { x = 2 }if (this.arr == 32) { x = 3 }if (this.arr >= 64) { x = 4 }// console.log(x)this.hand.destroy();//新手指引销毁if(!playerP){//如果是新用户var p = 1;cc.sys.localStorage.setItem('oldPlayer',p);};var playerP = cc.sys.localStorage.getItem('oldPlayer');// console.log(playerP);}},
onTouchMove(e) {if (this.state == 1) {// console.log("move");//区域控制this.scene();this.plane.setPosition(e.getDelta());//positionthis.ballNode.setPosition(e.getDelta());this.ballNode.y = -44}},

3.2:scene()函数

scene() {let PX = this.player.position.x;let PY = this.player.position.y;let BX = (cc.find("bg"), this.node).width / 2 - this.player.width / 2;if (PX < -BX) {this.player.runAction(cc.moveTo(0, -BX, PY));}if (PX > BX) {this.player.runAction(cc.moveTo(0, BX, PY));}},

4:小球合成

在小球碰撞合成的过程中,我使用的是引擎自带的碰撞组件(强烈不建议),建议各位开发者能自己写碰撞的一定要自己写,由于我自己还是学生(还很菜,不会自己写碰撞组件),所以就只能用引擎自带的了,这也就导致游戏存在一些问题,比如当场景中小球过多会引起“抖动”,小球合成视觉效果差(可能也和我没有美术有关,哈哈哈哈哈哈哈哈哈),各位开发者可以试着将它做成一种融合效果。言归正传,在这里,是进行判断两个碰撞体上Label节点上的数字是否相等,相等则进行小球合成操作,分数进行累加,小球合成以及消失动画(没有美术,动画有亿点点丑)。然后这里还存在一个小球合成后,小球的大小,颜色要进行更新,以及若合成的小球是场中最大的球,要将这个球的数据保存下来(后续要进行小球生成机制)。

onBeginContact: function (contact, selfCollider, otherCollider) {// console.log('begin')if(otherCollider.node.group == 'default'){let selfNum = selfCollider.node.getChildByName('thisNum').getComponent(cc.Label)// let self = selfCollider.node.getPosition();// console.log(self)let otherNum = otherCollider.node.getChildByName('thisNum').getComponent(cc.Label)var selfAnim = selfCollider.node.getComponent(cc.Animation);var otherAnim = otherCollider.node.getComponent(cc.Animation);//小球累加if(selfNum.string == otherNum.string){selfNum.string = selfNum.string*2//碰撞球数据更新// otherNum.string = 0;//被碰撞球数据清零//分数scorevar score = cc.find("Canvas/bg/score/scoreLabel").getComponent(cc.Label)this.s=selfNum.stringscore.string=Number(this.s)+Number(score.string)/*var gameScript = cc.find("Canvas/bg").getComponent('game')gameScript.updateScore(selfNum.string);*///最大数据更新/存入最大数据this.max = Math.max.apply(null,gNumArr)if(Number(this.s)>this.max){gNumArr.push(Number(this.s))}/*** 2048销毁*/if(Number(this.s)==2048){this.node.destroy();// boomNum+=1;}//Scale大小,颜色for(var i = 0;i<gBall.length;i++){if(selfNum.string==gBall[i].pro){this.node.setScale(gBall[i].scale)this.node.color=gBall[i].color;}};cc.audioEngine.playEffect(this.toucheffect,false);//动画播放selfAnim.playAdditive('self')otherAnim.playAdditive('other')// otherCollider.destroy()setTimeout(() => {otherCollider.node.destroy()}, 300);}}},

5:球大小及颜色的操作

球的大小和颜色这里用一个全局数组进行保存,然后通过小球Label组件上的数据和数组中的pro进行比较,然后获得颜色和大小。

5.1:数组代码

window.gBall=[{pro:2,color:new cc.Color(129,236,236),scale:0.55},{pro:4,color:new cc.Color(0, 206, 201),scale:0.6},{pro:8,color:new cc.Color(122,255,116),scale:0.65},{pro:16,color:new cc.Color(162,155,254),scale:0.7},{pro:32,color:new cc.Color(108,92,231),scale:0.75},{pro:64,color:new cc.Color(255,116,224),scale:0.8},{pro:128,color:new cc.Color(255,116,160),scale:0.85},{pro:256,color:new cc.Color(255,116,116),scale:0.9},{pro:512,color:new cc.Color(255,39,39),scale:0.95},{pro:1024,color:new cc.Color(16, 172, 132),scale:1.0},{pro:2048,color:new cc.Color(150,89,60),scale:1.05},
]

5.2:合成的主要部分

          //Scale/colorfor(var i = 0;i<gBall.length;i++){if(selfNum.string==gBall[i].pro){this.node.setScale(gBall[i].scale);//Scalethis.node.color=gBall[i].color;//color}};

6:球生成机制

在飞船移动生成小球的时候,存在一定的生成机制(当场上存在8才会生成小球4,存在16才会生成小球8.........),即需要判断当前场上存在的最大的小球的Label数据(生成即存放),然后才能随机生成新数据的小球。数据存入时一定存在一个判断,否则数组数据会非常多,不利于小球生成的数据获取。

6.1:数据存入

                //最大数据更新/存入最大数据this.max = Math.max.apply(null,gNumArr)//获取数组最大数据if(Number(this.s)>this.max){//若最大数据小于当前生成球的数据gNumArr.push(Number(this.s))//将当前球的数据存入数组}

6.2:小球生成机制

        //获取最大的数值for (var a = 0; a < gNumArr.length; a++) {this.arr = Math.max.apply(null, gNumArr)//获取数组内最大数据}// console.log(gNumArr)// console.log(this.arr)if (this.arr == 8) { x = 1 }if (this.arr == 16) { x = 2 }if (this.arr == 32) { x = 3 }if (this.arr >= 64) { x = 4 }

6.3:小球生成

updatestr(str){// console.log('data')this.Num.string = strfor(var a = 0;a<gBall.length;a++){if(this.Num.string==gBall[a].pro){this.node.setScale(gBall[a].scale)this.node.color=gBall[a].color;}};},

使用math函数求得(x+1)的2次方即可完成小球生成机制。

7:分数保存

在死亡(小球超出一定高度)后,游戏会进行分数保存,将用户所得的分数保存到本地并显示当前用户的最高分,这里不做过多解释,就是一个数据保存以及基本的判断。

saveScore() {//当前分数var currentScore = this.score.string;var scoreData = {'score': currentScore};var preData = cc.sys.localStorage.getItem('score');//获取数据var highScore = cc.sys.localStorage.getItem('topScore');//获取数据if (!highScore || parseInt(highScore) < parseInt(currentScore)) {//若不存在或最高分小于当前分数cc.sys.localStorage.setItem('topScore', currentScore);//存入数据}if (!preData) {preData = [];preData.unshift(scoreData);} else {preData = JSON.parse(preData);if (!(preData instanceof Array)) {preData = [];}preData.unshift(scoreData);}cc.sys.localStorage.setItem('currentScore', currentScore);//存入cc.sys.localStorage.setItem('score', JSON.stringify(preData));//存入console.log(preData)console.log(highScore)if (highScore < currentScore) {highScore = currentScore;}//最高分this.topScore.string = highScore;preTopScore = highScore;//当前分this.nowScore.string = currentScore;},

8:道具

最后一个也就是我们的道具,炸弹道具,道具的主要功能即当小球高度偏高时销毁屏幕中所有的小球,功能较为简单(也可以加入其他道具,如万能小球(可以与任意球合成)扔掉当前小球(随机新生成一个小球)),在这里程序部分也比较简单,即销毁其中一个节点下的所有子节点。由于这里我没有其他的炸弹生成机制,因此,道具只存在1个或0个的情况,如果想把生成2048后获得一个炸弹,这里可以做成自加自减的方式。

boomTouch() {if (this.state == 1) {if (boomNum > 0) {console.log('boom')boomNum = 0;// boomNum-=1;this.ballNodeDown.destroyAllChildren()} else {this.videoPlay()console.log('视频广告')}}},

 

这篇关于大三学生的游戏开发之旅-Cocos Creator微信小游戏实战(球球合成-2048)项目开发思路解析/方案的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/284208

相关文章

防止Linux rm命令误操作的多场景防护方案与实践

《防止Linuxrm命令误操作的多场景防护方案与实践》在Linux系统中,rm命令是删除文件和目录的高效工具,但一旦误操作,如执行rm-rf/或rm-rf/*,极易导致系统数据灾难,本文针对不同场景... 目录引言理解 rm 命令及误操作风险rm 命令基础常见误操作案例防护方案使用 rm编程 别名及安全删除

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Python实现批量CSV转Excel的高性能处理方案

《Python实现批量CSV转Excel的高性能处理方案》在日常办公中,我们经常需要将CSV格式的数据转换为Excel文件,本文将介绍一个基于Python的高性能解决方案,感兴趣的小伙伴可以跟随小编一... 目录一、场景需求二、技术方案三、核心代码四、批量处理方案五、性能优化六、使用示例完整代码七、小结一、

C#使用Spire.Doc for .NET实现HTML转Word的高效方案

《C#使用Spire.Docfor.NET实现HTML转Word的高效方案》在Web开发中,HTML内容的生成与处理是高频需求,然而,当用户需要将HTML页面或动态生成的HTML字符串转换为Wor... 目录引言一、html转Word的典型场景与挑战二、用 Spire.Doc 实现 HTML 转 Word1

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

使用docker搭建嵌入式Linux开发环境

《使用docker搭建嵌入式Linux开发环境》本文主要介绍了使用docker搭建嵌入式Linux开发环境,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录1、前言2、安装docker3、编写容器管理脚本4、创建容器1、前言在日常开发全志、rk等不同

使用Python实现Word文档的自动化对比方案

《使用Python实现Word文档的自动化对比方案》我们经常需要比较两个Word文档的版本差异,无论是合同修订、论文修改还是代码文档更新,人工比对不仅效率低下,还容易遗漏关键改动,下面通过一个实际案例... 目录引言一、使用python-docx库解析文档结构二、使用difflib进行差异比对三、高级对比方

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

Three.js构建一个 3D 商品展示空间完整实战项目

《Three.js构建一个3D商品展示空间完整实战项目》Three.js是一个强大的JavaScript库,专用于在Web浏览器中创建3D图形,:本文主要介绍Three.js构建一个3D商品展... 目录引言项目核心技术1. 项目架构与资源组织2. 多模型切换、交互热点绑定3. 移动端适配与帧率优化4. 可

深度解析Java @Serial 注解及常见错误案例

《深度解析Java@Serial注解及常见错误案例》Java14引入@Serial注解,用于编译时校验序列化成员,替代传统方式解决运行时错误,适用于Serializable类的方法/字段,需注意签... 目录Java @Serial 注解深度解析1. 注解本质2. 核心作用(1) 主要用途(2) 适用位置3