大三学生的游戏开发之旅-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

相关文章

解决Maven项目idea找不到本地仓库jar包问题以及使用mvn install:install-file

《解决Maven项目idea找不到本地仓库jar包问题以及使用mvninstall:install-file》:本文主要介绍解决Maven项目idea找不到本地仓库jar包问题以及使用mvnin... 目录Maven项目idea找不到本地仓库jar包以及使用mvn install:install-file基

Python使用getopt处理命令行参数示例解析(最佳实践)

《Python使用getopt处理命令行参数示例解析(最佳实践)》getopt模块是Python标准库中一个简单但强大的命令行参数处理工具,它特别适合那些需要快速实现基本命令行参数解析的场景,或者需要... 目录为什么需要处理命令行参数?getopt模块基础实际应用示例与其他参数处理方式的比较常见问http

Java Response返回值的最佳处理方案

《JavaResponse返回值的最佳处理方案》在开发Web应用程序时,我们经常需要通过HTTP请求从服务器获取响应数据,这些数据可以是JSON、XML、甚至是文件,本篇文章将详细解析Java中处理... 目录摘要概述核心问题:关键技术点:源码解析示例 1:使用HttpURLConnection获取Resp

Python利用ElementTree实现快速解析XML文件

《Python利用ElementTree实现快速解析XML文件》ElementTree是Python标准库的一部分,而且是Python标准库中用于解析和操作XML数据的模块,下面小编就来和大家详细讲讲... 目录一、XML文件解析到底有多重要二、ElementTree快速入门1. 加载XML的两种方式2.

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

java解析jwt中的payload的用法

《java解析jwt中的payload的用法》:本文主要介绍java解析jwt中的payload的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java解析jwt中的payload1. 使用 jjwt 库步骤 1:添加依赖步骤 2:解析 JWT2. 使用 N

springboot项目如何开启https服务

《springboot项目如何开启https服务》:本文主要介绍springboot项目如何开启https服务方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录springboot项目开启https服务1. 生成SSL证书密钥库使用keytool生成自签名证书将

Java实现优雅日期处理的方案详解

《Java实现优雅日期处理的方案详解》在我们的日常工作中,需要经常处理各种格式,各种类似的的日期或者时间,下面我们就来看看如何使用java处理这样的日期问题吧,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言一、日期的坑1.1 日期格式化陷阱1.2 时区转换二、优雅方案的进阶之路2.1 线程安全重构2

Python中__init__方法使用的深度解析

《Python中__init__方法使用的深度解析》在Python的面向对象编程(OOP)体系中,__init__方法如同建造房屋时的奠基仪式——它定义了对象诞生时的初始状态,下面我们就来深入了解下_... 目录一、__init__的基因图谱二、初始化过程的魔法时刻继承链中的初始化顺序self参数的奥秘默认

将Java项目提交到云服务器的流程步骤

《将Java项目提交到云服务器的流程步骤》所谓将项目提交到云服务器即将你的项目打成一个jar包然后提交到云服务器即可,因此我们需要准备服务器环境为:Linux+JDK+MariDB(MySQL)+Gi... 目录1. 安装 jdk1.1 查看 jdk 版本1.2 下载 jdk2. 安装 mariadb(my