您的位置:首页 > 移动开发 > Cocos引擎

cocos2d-x游戏开发(十)执行单元场景CCScene

2015-07-16 15:49 302 查看
欢迎转载:http://blog.csdn.net/fylz1125/article/details/8524081

游戏逻辑其实就像拍摄一部电影,整个过程都由导演来驱动。

游戏中的一个一个关卡,一个一个场景都跟电影中类似。每一个场景都有一些特定的元素,比如林冲风雪山神庙这个场景,大环境就是一个大风雪的夜晚,一个银装素裹的破败山头,有一个破败的山神庙,当然还可以有一些其他的元素,比如火光,野狼,灯光, 乌鸦连声叫、黄狗大声吼等。这就是一个特定的场景了,呵呵。然后导演大叫一声:“Action!”,然后所有的故事情节就在这里展开,主角和一干配角上场了...

以上所有的元素就组成了这么一个场景,那么在游戏里又如何组织这些元素呢。先来看看cocos2d-x的场景处理流程。

cocos2d-x中的导演执行单元就是场景CCScene。一个场景又有那么多元素,比如一个大的背景,然后又有一些小元素,比如星星,火堆,狗等,又是如何加到场景里的呢。

可以这么说,一个场景就是一个特别的容器(其实就是一个CCNode),它能容纳各类节点(通过addChild)。那么跟电影一样,镜头一出来先有一个大的环境。这个就是主背景层(CCLayer),对,就是CCLayer(怎么翻译好一点)。然后在这个主Layer里面再添加一些其他的元素,比如人物、火堆、狂奔的大狗、清唱的小鸟等。这些元素都是独立的对象(继承CCSprite),他们都有各自的属性和行为,比如人物是一个男人,他在漫无目的的行走,火堆在燃烧放出一跳一跳的火光,狗在奔跑,鸟在飞来飞去等。所有这些元素都添加到主Layer里面,然后这个主Layer又被添加都场景里面(CCScene),最后导演执行这个场景(pDirector->runWithScene(pScene)),一个画面就出来了。

来看一看代码级别的执行过程:

主循环(不知道的看我前面的文章)

[cpp] view
plaincopyprint?

void CCDisplayLinkDirector::mainLoop(void)

{

if (m_bPurgeDirecotorInNextLoop)

{

m_bPurgeDirecotorInNextLoop = false;

purgeDirector();

}

else if (! m_bInvalid)

{

drawScene();// 注意这里就是画场景了



// release the objects

CCPoolManager::sharedPoolManager()->pop();

}

}

跟进drawScene()看看怎么画的

[cpp] view
plaincopyprint?

void CCDirector::drawScene(void)

{

// calculate "global" dt

calculateDeltaTime();



//tick before glClear: issue #533

if (! m_bPaused)

{

m_pScheduler->update(m_fDeltaTime);//按照优先级调度update函数(每个节点都有这么个函数)

}



glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);



/* to avoid flickr, nextScene MUST be here: after tick and before draw.

XXX: Which bug is this one. It seems that it can't be reproduced with v0.9 */

if (m_pNextScene) // 这个变量存下次绘制的场景,本次绘制则是上一次设置的场景

{

setNextScene();// 注意这个,不要被名字误导,其实就是设置下一次要运行的场景

}



kmGLPushMatrix();



// draw the scene

if (m_pRunningScene)

{

m_pRunningScene->visit();// 注意看这里,这个visit函数就是递归绘制子节点

}



// draw the notifications node

if (m_pNotificationNode)

{

m_pNotificationNode->visit();

}



if (m_bDisplayStats)

{

showStats();

}



kmGLPopMatrix();



m_uTotalFrames++;



// swap buffers

if (m_pobOpenGLView)

{

m_pobOpenGLView->swapBuffers();

}



if (m_bDisplayStats)

{

calculateMPF();

}

}

好,看注释,有个visit函数,这个就是递归绘制子节点的

[cpp] view
plaincopyprint?

void CCNode::visit()

{

// quick return if not visible. children won't be drawn.

if (!m_bVisible)

{

return;

}

kmGLPushMatrix();



if (m_pGrid && m_pGrid->isActive())

{

m_pGrid->beforeDraw();

}



this->transform();



CCNode* pNode = NULL;

unsigned int i = 0;



if(m_pChildren && m_pChildren->count() > 0)//这里开始,如果有子节点就进入

{

sortAllChildren();// 按z坐标排序,就是z序排列子节点

// draw children zOrder < 0

ccArray *arrayData = m_pChildren->data;

for( ; i < arrayData->num; i++ )//画z序<0的子节点

{

pNode = (CCNode*) arrayData->arr[i];



if ( pNode && pNode->m_nZOrder < 0 )

{

pNode->visit();

}

else

{

break;

}

}

// self draw

this->draw(); // 画自己



for( ; i < arrayData->num; i++ )

{

pNode = (CCNode*) arrayData->arr[i];

if (pNode)

{

pNode->visit();

}

}

}

else

{

this->draw(); // 没有子节点就画自己

}



// reset for next frame

m_uOrderOfArrival = 0;



if (m_pGrid && m_pGrid->isActive())

{

m_pGrid->afterDraw(this);

}



kmGLPopMatrix();

}

这个是主循环的,但是要运行一个场景是从哪开始的呢,比如

[cpp] view
plaincopyprint?

CCScene *pScene = HelloWorld::scene();



// run

pDirector->runWithScene(pScene);

看,第一个场景从这里开始的,跟进

[cpp] view
plaincopyprint?

void CCDirector::runWithScene(CCScene *pScene)

{

CCAssert(pScene != NULL, "This command can only be used to start the CCDirector. There is already a scene present.");

CCAssert(m_pRunningScene == NULL, "m_pRunningScene should be null");// 当前运行为NULL(第一次)



pushScene(pScene);// push是什么呢

startAnimation(); // 这里控制帧率,同时又进入主循环

}

看这个pushScene函数,是干什么的

[cpp] view
plaincopyprint?

void CCDirector::pushScene(CCScene *pScene)

{

CCAssert(pScene, "the scene should not null");



m_bSendCleanupToScene = false;



m_pobScenesStack->addObject(pScene);// 当前scene入栈

m_pNextScene = pScene; // next Scene,就是下一次绘制当前scene

}

好了,这个pushScene干了两件事,将要绘制的scene入栈,将下次要绘制的场景设为pScene。这个函数返回后接着是

startAnimation(),这个函数调整帧率然后再次进入主循环。

再回头看看主循环的drawScene()函数(上面有),其中有段:

[cpp] view
plaincopyprint?

if (m_pNextScene)// 刚才的pushScene函数设置了这个值

{

setNextScene(); // 将下个要运行的Scene给m_pRunningScene

}



kmGLPushMatrix();



// draw the scene

if (m_pRunningScene) // 这里开始渲染当前要运行的Scene

{

m_pRunningScene->visit();

}

看代码有点绕,其实就是一个迭代过程。最开始第一次m_pRunningScene是NULL,然后 runWithScene后就有了,m_pRunningScene就是你传进去的Scene,不过是在下次循环绘制运行。这里有两个过程,一个就是runWithScene函数和replaceScene函数,这两个函数都会设置下个Scene,同时将要运行的Scene入栈。然后进入主循环,在主循环中每次都会检测当前要运行的Scene,如果有就渲染执行。就是这么一个过程。

导演每次只能执行一个场景,执行并不仅仅是绘制,还包括其动作,回调,过渡等。每个场景实质上是一个CCNode,

前面讲过,CCNode有个很重要的成员变量m_pChildren,这是一个CCArray,用来存放添加到该节点的子节点。Scene在渲染的时候会递归渲染其子节点,这就组成一个渲染链,保证整个场景的所有元素都会被渲染。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: