大三学生的游戏开发之旅-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)项目开发思路解析/方案的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:https://blog.csdn.net/qq_45310244/article/details/106899133
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/284208

相关文章

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

Java MQTT实战应用

《JavaMQTT实战应用》本文详解MQTT协议,涵盖其发布/订阅机制、低功耗高效特性、三种服务质量等级(QoS0/1/2),以及客户端、代理、主题的核心概念,最后提供Linux部署教程、Sprin... 目录一、MQTT协议二、MQTT优点三、三种服务质量等级四、客户端、代理、主题1. 客户端(Clien

在Spring Boot中集成RabbitMQ的实战记录

《在SpringBoot中集成RabbitMQ的实战记录》本文介绍SpringBoot集成RabbitMQ的步骤,涵盖配置连接、消息发送与接收,并对比两种定义Exchange与队列的方式:手动声明(... 目录前言准备工作1. 安装 RabbitMQ2. 消息发送者(Producer)配置1. 创建 Spr

SQLite3 在嵌入式C环境中存储音频/视频文件的最优方案

《SQLite3在嵌入式C环境中存储音频/视频文件的最优方案》本文探讨了SQLite3在嵌入式C环境中存储音视频文件的优化方案,推荐采用文件路径存储结合元数据管理,兼顾效率与资源限制,小文件可使用B... 目录SQLite3 在嵌入式C环境中存储音频/视频文件的专业方案一、存储策略选择1. 直接存储 vs

使用Python绘制3D堆叠条形图全解析

《使用Python绘制3D堆叠条形图全解析》在数据可视化的工具箱里,3D图表总能带来眼前一亮的效果,本文就来和大家聊聊如何使用Python实现绘制3D堆叠条形图,感兴趣的小伙伴可以了解下... 目录为什么选择 3D 堆叠条形图代码实现:从数据到 3D 世界的搭建核心代码逐行解析细节优化应用场景:3D 堆叠图

深度解析Python装饰器常见用法与进阶技巧

《深度解析Python装饰器常见用法与进阶技巧》Python装饰器(Decorator)是提升代码可读性与复用性的强大工具,本文将深入解析Python装饰器的原理,常见用法,进阶技巧与最佳实践,希望可... 目录装饰器的基本原理函数装饰器的常见用法带参数的装饰器类装饰器与方法装饰器装饰器的嵌套与组合进阶技巧

java向微信服务号发送消息的完整步骤实例

《java向微信服务号发送消息的完整步骤实例》:本文主要介绍java向微信服务号发送消息的相关资料,包括申请测试号获取appID/appsecret、关注公众号获取openID、配置消息模板及代码... 目录步骤1. 申请测试系统2. 公众号账号信息3. 关注测试号二维码4. 消息模板接口5. Java测试