1. 游戏视频



主角眼熟吗?没错,上次跑酷游戏中的“30”来Jump了,有三种道具,主角光环,竹蜻蜓,翅膀; 有两种怪物,螃蟹和鸟; 有5种板子。点击屏幕,30会把它的嘴巴3给发射出去,可以攻击怪物。上次的跑酷参考​​《​​​​cocos2d 简单高仿天天跑酷游戏》​​ ,苹果已经审核通过...但做的很差,后续有时间再更新。后面考虑会出更多的该主角游戏。

 

cocos2d 高仿doodle jump 无源码_2d


其实蛮讨厌"高仿"这两个字的,争取下一款游戏有更多原创部分。


2.向上跳的位移实现


cocos2d 高仿doodle jump 无源码_2d_02


正常情况下,不使用任何的东西,主角在板子上跳,会跳出最上面的屏幕就看不到了!没有视频中主角一直在屏幕中间的感觉。那么如何实现这样的效果呢?非常简单。

这里不得不吐槽下cocos2d, 学过Opengl 或者 DirectX的,都应该知道有Camera这么一个东西存在,2D游戏只是选择了平行投射(parallel projection)参考这篇文章​​《DirectX Camera》​​ .但3.2 之前的Camera根本不能用,3.3才加入了Camera. 如果用Camera这里会非常容易实现,也跟现实比较吻合。

1.如果使用Camera,所有板的PositionY 就定死的,越高的板,它的PositionY 就越大,主角往上跳时,主角的PositionY也不断变大。有一个Camera,它的位置跟着主角来移动就可以了。就这么简单。

2.不使用Camera,我用的是3.2的cocos2d,所以不好使用Camera,采用了移动整个Layer的实现。主角会跳出最上面的屏幕的,怎么办的?把整个Layer往下移动就可以了。

float minPositionY = 0;
void update(float dt){
//follow the hero
float nextPositionY = VISIBLE_HEIGHT / 2 - hero->getPositionY();
if(nextPositionY < minPositionY){
minPositionY = nextPositionY;
this->setPositionY(minPositionY);
}
}


看起来比较简单,如果是主角掉下屏幕最底下,有一个坠落感觉,逻辑上又比较麻烦了。还是用Camera的比较好。


3.板子生成的逻辑


最块又比较麻烦了。首先是板子连续生成的逻辑,不可能一次生成几十个板子,反正屏幕之上和屏幕之下用户都是看不到的,也就是说10几个板子,整个游戏就够用了。到后面,10个都不用。正常的update是一秒中执行60次。我们需要整一个慢一点的定时器,这样不会太影响性能, schedule ( schedule_selector ( GameObjectsLayer :: createAndRemoveObjects ), 0.064 );,专门在里面创建板和移除不需要的板子。

deque<JumpBoard*> _boardObjects;
void generateSomeBoards(const int& lastPositionX,int startPosition,const int& count);
void createAndRemoveObjects(float dt){
if(_pause == true){
return;
}
if(_boardObjects.size() <= 12){//generate some boards if we only have few boards
JumpBoard *lastBoard = _boardObjects.back();
generateSomeBoards(lastBoard->getPositionX(), lastBoard->getPositionY() + getGeneratorHigh(),5);
}
//remove unused boards
JumpBoard* firstBoard = _boardObjects.front();
if(hero->getPositionY() - firstBoard->getPositionY() > 600){
firstBoard->removeFromParent();
_boardObjects.pop_front();
}

}


再简单用语言来说下上面中没有给出详细代码的generateSomeBoards。

游戏要考虑的东西很多:

1.刚开始的时候两个板之间的距离要短些,看起来会密些,这样游戏会容易上手些,两个板之间的距离会随着主角的变高而增大。

2.刚开始的时候生成板的类型要简单些,我这里只有最两种类型,到后面才会出现各种类型的板。

我现在越来越意识到,如果一个游戏的主要玩法能让不懂程序逻辑的策划很容易修改的话,这个游戏写的就还不错。这里游戏的玩法就是如何控制生成板子。我必须承认这块做的还不好。


typedef enum{
kNormalBoard, //0
kCloudBoard, //1
kMoveBoard, //2
kMoveLttleBoard, //3
kSpringBoard //4
}JumpBoardType;


int level1[] = {0,0,0,0,0,0,0,0,0,0,1,1,1};
int level2[] = {0,0,0,0,0,0,0,0,0,0,1,1,2,3,4};


这里简单有两个数组,后面会用它们初始化两个vector<int>,然后用random_shuffle,进行洗牌。会根据当前的hero高度来选择不同的数组。当index移到最后一个时候,就归0,再次洗牌。


int getBoardTypeDueToFloorNum(){
int floorCount = getFloorCount();
int result = 0;
if(floorCount <= 2){
if(_levelIndex >= _level1.size()){
_levelIndex = 0;
random_shuffle(_level1.begin(), _level1.end());
}
result = _level1[_levelIndex];
++_levelIndex;
return result;

}else if(floorCount > 2 && floorCount <= 10){
if(_levelIndex >= _level2.size()){
_levelIndex = 0;
random_shuffle(_level2.begin(), _level2.end());
}
result = _level2[_levelIndex];
++_levelIndex;
return result;
}else{
//无限模式
}
}


当然后面的无限模式更加复杂点,我搞了5种不同类型的数组进行洗牌。这里不再赘述。


4.道具和怪物生成位置


道具和怪物生成位置我就简单处理了,每隔几个屏幕高度就放置一个。

//typedef enum{
// kHelicopterItem, //0
// kFlyItem, //1
// kShortItem, //2
// kLongItem, //3
// kDefenceBall //4
//}EatItemType;

int allEatItemType[] = {4,4,4,0,0,1};


为了让不同道具出现概率不一样,又用了跟上面板子类似的逻辑。怪物这里就2种,就50%概率会生成不同怪物。Doodle jump这块做的变态的好,一般怪物出现之前会有杀死怪物或者躲避怪物的道具出现。


5.主角光环和抛物线掉落


主角光环的椭圆移动,可以参考这篇文章​​《​​​ ​​cocos2d 绕椭圆移动》​​。

还有一个是使用过后道具的掉落轨迹。这块第一次玩Doodle jump就给我带来了惊喜,竹蜻蜓或者火箭,时间到后,完美的扔掉,处理的非常自然和有趣。

实现起来跟上面的椭圆移动类似。就是数学用的抛物线,只是这个是开口向下的。只用到了左边红色的部分。

我用的公式是x = t * -400,  y = x * x *  - 0.02 ( x  = [0, -400] )。update中,t = [0, 1],代入这个公式就可以了。

cocos2d 高仿doodle jump 无源码_跑酷游戏_03


6. 其他


其他像道具生成的位置比较讲究,最后添加到板子中去,这样板子移动,就会跟着移动了,会自然点。

无论是主角碰上怪物或者主角的嘴巴碰上怪物,怪物都会旋转地往那个方向飞出去,这个效果非常好,可以简单地计算下两个点的角度,给怪物一个速度属性,在update的每一帧中进行移动。

主角的嘴巴飞出去还会飞回来,飞回来的时候,因为主角的位置一直变化的,所以也要在update的每一帧中,不断改变位置,离主角越来越近,像飞机的跟踪导弹。这里就不再赘述。


感觉Cocos2d-x 3.2这块重力感应Accelerometer控制的似乎有问题,在这个游戏中偶尔会失控,我还不知道发生的原因和规律,非常头痛。还有游戏的速度,看起来都是60帧,但速度偶尔会有区别。

这些高仿的游戏我想不太可能有好的市场,只能通过它们积累经验来做真正原创游戏。不过做这些小游戏比那些网络游戏有趣些。有真正游戏的感觉。