码迷,mamicode.com
首页 > 其他好文 > 详细

Cocos2d-x 3.4 之 消灭星星 > 第二篇 <

时间:2015-05-24 23:39:21      阅读:215      评论:0      收藏:0      [点我收藏+]

标签:cocos2d-x   3.4   消灭星星   第二篇   

***************************************转载请注明出处:http://blog.csdn.net/lttree********************************************





这个,要说一下,

本系列文章,我是边写边发,所以可能前后有些改动。


上篇文章,已经做到了  游戏场景,点击星星,使其高亮状态。

这篇文章,将游戏逻辑玩法搞定:

> 星星 高亮状态 再次点击,消除

> 星星消除后,剩下的星星整合(下落或左移)

> 检测是否有可消除的星星,没有则跳转到 游戏结束界面



正文:


1. 星星 高亮状态下的消除

触摸事件函数 onTouchBegan需要重写一下:

// 触摸事件
bool GameScene::onTouchBegan(Touch *touch, Event *unused)	{
	
	auto location = touch->getLocation();
	Star* sta;
	sta = starOfPoint(&location);
	
	// 当前对象不为空
	if( sta )	{
		// 看该 星星 处于什么状态
		if( sta->getDisplayMode() == DISPLAY_MODE_NORMAL )	{

			// 初始化记忆地图mapR+恢复普通状态
			for( int i = 0 ; i < ROWS ; i++ )	{
				for( int j = 0 ; j <COLS ; j++ )	{
					mapR[i][j] = false;
					if( map[i][j]!=NULL )
						map[i][j]->setDisplayMode(DISPLAY_MODE_NORMAL);
				}
			}

			// 如果只有当前一个星星这个颜色,不要让它处于高亮状态
			if( waitPop(sta->getRow(),sta->getCol()) == 1 )	{
				mapR[sta->getRow()][sta->getCol()]=false;
				sta->setDisplayMode(DISPLAY_MODE_NORMAL);
			}
		}
		else
		{
			deleteStar();
		}
		
	}

	return true;
}

在 所触摸的星星 非空的前提下,

判断该星星是否为高亮状态,如果不是,则初始化mapR,并将所有 星星 的显示状态恢复成 普通,

注意,这里所有星星的显示状态中,要把消除掉的过滤掉,否则会出错的。

然后是 deleteStar函数:

void GameScene::deleteStar( void )
{
	int r,c;
	for( r = 0 ; r < ROWS ; r++ )	{
		for( c = 0 ;c < COLS ; c++ )	{
			if( mapR[r][c] == true )	{	
				// popStar(map[i][j]);

				map[r][c]->removeFromParent();
				map[r][c]=NULL;
			}
		}
	}

}

这里的popStar函数,是用来后期加一些特效啥的,先保留一下。

注意,在删除某个星星的时候,不要忘了 removeFromParent,将该对象从父节点拿掉。


还有一点,在本类构造函数时候,要对 mapR数组初始化,

可以用 memset,也可以两个for循环,甚至你写N个false赋值也没问题...

这里我用的memset,但要包含 string.h 头文件

GameScene::GameScene()	
: starSheet(NULL)
,mapLBX ( (GAME_SCREEN_WIDTH - STAR_WIDTH * COLS - (COLS - 1) * BOADER_WIDTH) / 2 )
,mapLBY ( (GAME_SCREEN_HEIGHT - STAR_WIDTH * ROWS - (ROWS - 1) * BOADER_WIDTH) / 2 )
{
	// 初始化 mapR 数组(string.h 头文件)
	memset(mapR,false,sizeof(mapR));
}

现在,我们就可以将高亮的 星星 删除掉了

演示一下:(PS:一直想用GIF演示,但是没找到合适的GIF录制工具(GIFCAM不好用),求大家推荐一款)

技术分享                         技术分享




2.消除后的整合

在将星星消除后,我们要将剩余的星星聚起来,

所有的星星就是向左下角移动,先向下,再向左....

说一下思路:

向下归拢: 按列为单位,每列中从下往上找,

若该对象为空,将上面第一个不为空的,落下来;

然后再向上找,直到最后。

向左归拢:依旧按列为单位,先判断每列第一行是否为空,

若为空,需要将右面的平移动过来,(因为先下落,再左移,如果第一行为空,上面肯定也为空,所以可以平移)

要注意,这个与下落不一样,每列向左平移相同距离(不是补齐)

Ok,实现起来就是这样:

void GameScene::adjustStar( )
{
	int r,c;
	// 向下归拢
	for( c = 0 ; c < COLS ; c++ )	{
		for( r = 0 ; r < ROWS-1 ; r++ )	{
			if( map[r][c] == NULL )	{
				for( int newR = r+1 ; newR < ROWS ; newR++ )	{
					Star* sta = map[newR][c];
					if( sta != NULL )	{

						map[r][c] = sta;
						map[newR][c] = NULL;

						Point startPosition = sta->getPosition();
						Point endPosition = positionOfItem(r, c);
						float speed = (startPosition.y - endPosition.y) / GAME_SCREEN_HEIGHT*2;
						sta->stopAllActions();
						sta->runAction(MoveTo::create(speed, endPosition));
						sta->setRow(r);

						break;
					}
				}
			}
		}
	}

	// 向左归拢
	for( c = 0 ; c < COLS-1 ; c++ )	
		if( map[0][c] == NULL )	{
			for( int newC=c+1; newC < COLS ; newC++ )	{
				if( map[0][newC] != NULL )	{
					for( int moveR = 0 ; moveR < ROWS ; moveR++ )	{
						Star* sta = map[moveR][newC];
						
						if( sta!=NULL )	{
							map[moveR][c] = sta;
							map[moveR][newC] = NULL;

							Point startPosition = sta->getPosition();
							Point endPosition = positionOfItem(moveR,c);
							float speed = (startPosition.x - endPosition.x) / GAME_SCREEN_WIDTH*2;
							sta->stopAllActions();
							sta->runAction(MoveTo::create(speed, endPosition));
							sta->setCol(c);
						}
					}
					break;
				}
			}
		}
}

一个非常重要一点,在runAction之前,一定要 stopAllActions!!!

就这一句,我调了一下午,~~o(>_<)o ~~

演示一下:(PS:一直想用GIF演示,但是没找到合适的GIF录制工具(GIFCAM不好用),求大家推荐一款)

技术分享                          技术分享




3.判断游戏是否结束

这个相对来说就简单很多,因为数组不大,直接遍历,

将 本对象 与上下左右四个方向上对象比较,

如果有同颜色,直接返回false(表示 未结束)

遍历结束后,若没有出现相同颜色的,返回true(表示 结束)

bool GameScene::isFinish( )
{
	int i,r,c;
	// 四个方向 
	int search[4][2]={-1,0,1,0,0,1,0,-1};

	for( r = 0 ; r < ROWS ; r++ )	{
		for( c = 0 ; c < COLS ; c++ )	{
			if( !map[r][c] )	continue;

			for( i = 0 ; i < 4 ; i++ )	{
				int nr = r + search[i][0];
				int nc = c + search[i][1];

				if( nr>=0 && nr<ROWS && nc>=0 && nc<COLS )	{
					if( map[nr][nc]!=NULL )	{
						if( map[r][c]->getImgIndex() == map[nr][nc]->getImgIndex() )
							return false;
					}
				}
			}
		}
	}

	return true;
}

接下来,简单设置一下,跳转什么的,就可以了。

技术分享

加新的场景(主界面场景)及跳转那些就不再列出了,直接看下面的源码吧。


游戏大体逻辑已经搞定了,

接下来就是一些增分性内容了,

分数、动画、还有我最想练的 粒子特效~



本篇文章源码:  > 这里 <



***************************************转载请注明出处:http://blog.csdn.net/lttree********************************************

Cocos2d-x 3.4 之 消灭星星 > 第二篇 <

标签:cocos2d-x   3.4   消灭星星   第二篇   

原文地址:http://blog.csdn.net/lttree/article/details/45920681

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!