CCScene一般情况是游戏里面的根节点,称之为"场景",运行游戏时需要通过CCDirector启动第一个场景。当然,游戏稍微复杂一点的话,可能会包含很多个场景,这就涉及到场景的切换,也是通过CCDirector来完成。CCScene是个抽象的概念,也没有可视化显示的功能,对比CCNode,CCScene基本上没有额外的代码:

  1. // CCScene.h
  2. #import "CCNode.h"
  3. @interface CCScene : CCNode
  4. {
  5. }
  6. @end


  1. // CCScene,m
  2. #import "CCScene.h"
  3. #import "Support/CGPointExtension.h"
  4. #import "CCDirector.h"
  5. @implementation CCScene
  6. -(id) init
  7. {
  8. if( (self=[super init]) ) {
  9. CGSize s = [[CCDirector sharedDirector] winSize];
  10. // 设置position不受anchorPoint影响
  11. self.isRelativeAnchorPoint = NO;
  12. // 设置anchorPoint
  13. anchorPoint_ = ccp(0.5f, 0.5f);
  14. // 设置CCScene的大小为屏幕大小
  15. [self setContentSize:s];
  16. }
  17. return self;
  18. }
  19. @end

可以发现,对比CCNode,CCScene只是重写了init方法而已,而且也只是做了一个简单的设置

常用操作

1.运行第一个场景

我们一般是在应用程序代理AppDelegate的applicationDidFinishLaunching:方法结尾处运行游戏的第一个场景

  1. [[CCDirector sharedDirector] runWithScene: [HelloWorldLayer scene]];

这里的[HelloWorldLayer scene]返回的是一个CCScene对象

2.替换场景

  1. [[CCDirector sharedDirector] replaceScene:scene];

这个方法会用新的场景替换旧的场景,cocos2d会释放旧场景的内存,删除旧场景中所有的节点,停止所有动作和消息调度,因此我们不用手动释放旧场景的内存

3.推入和弹出场景

我们知道可以用replaceScene:来运行一个新场景,但是会释放掉旧场景的内存。有时候我们希望在不释放旧场景内存的前提下运行一个新场景,这时候就要用到CCDirector的pushScene:和popScene两个方法了。

1> 使用pushScene:方法推入一个新场景,新场景会层叠在旧场景的上面,但并没有释放旧场景的内存,旧场景继续保留在内存中

  1. [[CCDirector sharedDirector] pushScene:scene];

2> 使用popScene方法弹出最上层的场景并释放其内存,使保留在内存中的旧场景重新显示出来

  1. [[CCDirector sharedDirector] popScene];

CCTransitionScene

上面介绍了场景的切换,不过都是瞬间完成的,有时候我们想在场景切换的时候有些过渡效果,即以动画的形式切换场景,我们称之为"场景过渡"。要想做场景过渡效果,就必须用CCTransitionScene的子类,CCTransitionScene本身继承了CCScene,它包含了非常多的子类,每个子类都有不同的场景过渡效果,比如CCTransitionFade是淡入淡出效果,CCTransitionPageTurn是翻页效果。

下面演示一个翻页效果:

  1. CCTransitionPageTurn *page = [CCTransitionPageTurn transitionWithDuration:0.5 scene:scene];
  2. [[CCDirector sharedDirector] replaceScene:page];

意思是在0.5秒的时间内使用翻页效果从旧场景过渡到scene这个新场景,因为CCTransitionScene是CCScene的子类,所以可以作为replaceScene:的参数。

cocos2d中有非常多的过渡效果可以使用,都是CCTransitionScene的子类,类名一般都是以CCTransition开头的。我就不在这里一一介绍每个子类有什么效果,也没有必要,用到时自己再去查API吧。

注意:CCTransitionScene只能使用在replaceScene:和pushScene:的时候,在popScene弹出场景时是不能用这个过渡效果的

节点的生命周期

说到场景过渡,那就不得不说一下节点的生命周期,即一个节点从开始被添加到屏幕上 到 从屏幕中移除的过程,CCNode提供了相应的生命周期方法:

  1. // 节点被添加到屏幕上  或者 重新显示到屏幕上 时调用
  2. -(void) onEnter;
  3. // 调用完onEnter后就会调用此方法,如果使用了场景过渡效果,将在场景过渡完毕后才调用此方法
  4. -(void) onEnterTransitionDidFinish;
  5. // 节点从屏幕中移除 或者 暂时离开屏幕 时调用
  6. -(void) onExit;

下面演示在场景切换时,节点生命周期方法的调用顺序

假设有2个图层RedLayer和BlueLayer,它们分别在不同的场景中。点击RedLayer,就推入BlueLayer所在的场景,点击BlueLayer就弹出BlueLayer所在的场景。我们就在场景切换的过程中观察这2个图层的生命周期。

为了区分这2个图层,我让它们继承了CCLayerColor,分别设置不用的背景颜色,RedLayer为红色,BlueLayer为蓝色。

因为RedLayer和BlueLayer都继承CCLayerColor,而且都需要负责创建自己的图层、负责观察生命周期方法的调用,那么我就先抽出一个继承了CCLayerColor的公共父类BaseLayer,在它里面完成一些公共操作,然后让RedLayer和BlueLayer都继承它

BaseLayer的代码

  1. // BaseLayer.h
  2. #import "cocos2d.h"
  3. @interface BaseLayer : CCLayerColor
  4. // 用来创建图层所在的场景
  5. + (CCScene *)scene;
  6. // 图层的背景颜色,交给子类去实现
  7. + (ccColor4B)bgColor;
  8. @end
  1. // BaseLayer.m
  2. #import "BaseLayer.h"
  3. @implementation BaseLayer
  4. #pragma mark - 初始化场景
  5. + (CCScene *)scene {
  6. // 获取当前类的背景颜色
  7. ccColor4B color = [self bgColor];
  8. // 根据当前类名创建图层
  9. BaseLayer *layer = [[self class] layerWithColor:color];
  10. // 接收触摸输入
  11. layer.isTouchEnabled = YES;
  12. CCScene *scene = [CCScene node];
  13. [scene addChild:layer];
  14. return scene;
  15. }
  16. #pragma mark - 打印生命周期方法
  17. - (void)onEnter { // _cmd 代表着当前的selector
  18. [super onEnter];
  19. // 第一个%@是打印类名,第二个%@是打印方法名
  20. NSLog(@"%@ --> %@", NSStringFromClass([self class]), NSStringFromSelector(_cmd));
  21. }
  22. - (void)onEnterTransitionDidFinish {
  23. [super onEnterTransitionDidFinish];
  24. NSLog(@"%@ --> %@", NSStringFromClass([self class]), NSStringFromSelector(_cmd));
  25. }
  26. - (void)onExit {
  27. [super onExit];
  28. NSLog(@"%@ --> %@", NSStringFromClass([self class]), NSStringFromSelector(_cmd));
  29. }
  30. @end

RedLayer的代码

  1. // RedLayer.h
  2. #import "BaseLayer.h"
  3. @interface RedLayer : BaseLayer
  4. @end
  1. // RedLayer.m
  2. #import "RedLayer.h"
  3. #import "BlueLayer.h"
  4. @implementation RedLayer
  5. #pragma mark - 点击红色图层时,跳到蓝色图层所在的场景
  6. - (void)ccTouchesEnded:(NSSet *)touches withEvent:(UIEvent *)event {
  7. // 初始化蓝色图层所在的场景
  8. CCScene *scene = [BlueLayer scene];
  9. // 推入场景(暂时没有使用过渡效果)
  10. [[CCDirector sharedDirector] pushScene:scene];
  11. }
  12. #pragma mark - 背景颜色为红色
  13. + (ccColor4B)bgColor {
  14. return ccc4(255, 0, 0, 255);
  15. }
  16. @end

BlueLayer的代码

  1. // BlueLayer.h
  2. #import "BaseLayer.h"
  3. @interface BlueLayer : BaseLayer
  4. @end
  1. // BlueLayer.m
  2. #import "BlueLayer.h"
  3. @implementation BlueLayer
  4. #pragma mark - 当点击蓝色图层时,弹出场景
  5. - (void)ccTouchesEnded:(NSSet *)touches withEvent:(UIEvent *)event {
  6. [[CCDirector sharedDirector] popScene];
  7. }
  8. #pragma mark - 背景颜色为蓝色
  9. + (ccColor4B)bgColor {
  10. return ccc4(0, 0, 255, 255);
  11. }
  12. @end

1.在应用程序加载完毕后,即在AppDelegate的applicationDidFinishLaunching:方法中启动第一个场景 ---- 红色图层所在的场景

  1. [[CCDirector sharedDirector] runWithScene: [RedLayer scene]];

运行完毕后,效果如下:

生命周期方法打印如下:

  1. 2013-02-22 15:47:56.473 HelloWorld[2679:c07] RedLayer --> onEnter
  2. 2013-02-22 15:47:56.474 HelloWorld[2679:c07] RedLayer --> onEnterTransitionDidFinish

2.点击红色图层,跳到蓝色图层所在的场景

这里根据有没有使用过渡效果,要分2种情况

1> 如果没有使用过渡效果

屏幕直接变为蓝色

生命周期方法打印如下:

  1. 2013-02-22 15:50:16.381 HelloWorld[2679:c07] RedLayer --> onExit
  2. 2013-02-22 15:50:16.382 HelloWorld[2679:c07] BlueLayer --> onEnter
  3. 2013-02-22 15:50:16.384 HelloWorld[2679:c07] BlueLayer --> onEnterTransitionDidFinish

可以看出,是先移除红色,再添加蓝色

2> 如果使用了过渡效果

先改变下RedLayer中的代码:

  1. #pragma mark - 点击红色图层时,跳到蓝色图层所在的场景
  2. - (void)ccTouchesEnded:(NSSet *)touches withEvent:(UIEvent *)event {
  3. // 初始化蓝色图层所在的场景
  4. CCScene *scene = [BlueLayer scene];
  5. // 旧场景一边旋转一边缩小,新场景一边旋转一边方法
  6. CCTransitionRotoZoom *page = [CCTransitionRotoZoom transitionWithDuration:2 scene:scene];
  7. // 推入场景(暂时没有使用过渡效果)
  8. [[CCDirector sharedDirector] pushScene:page];
  9. }

接下来看一下屏幕效果和打印信息

过渡效果刚开始就会打印:

  1. 2013-02-22 15:59:38.420 HelloWorld[2862:c07] BlueLayer --> onEnter

说明是先初始化并添加BlueLayer

然后中间经历了长达2s的过渡效果:

红色是在一边旋转一边缩小

红色完全消失后,蓝色在一边旋转一边放大

蓝色放大到屏幕大小后

最后会出现以下打印信息:

  1. 2013-02-22 16:04:28.852 HelloWorld[2910:c07] RedLayer --> onExit
  2. 2013-02-22 16:04:28.853 HelloWorld[2910:c07] BlueLayer --> onEnterTransitionDidFinish

移除红色,蓝色过渡完毕

3> 点击蓝色图层,弹出蓝色图层所在的场景,重新显示红色图层所在的场景

打印信息如下:

  1. 2013-02-22 16:06:47.013 HelloWorld[2910:c07] BlueLayer --> onExit
  2. 2013-02-22 16:06:47.014 HelloWorld[2910:c07] RedLayer --> onEnter
  3. 2013-02-22 16:06:47.015 HelloWorld[2910:c07] RedLayer --> onEnterTransitionDidFinish

移除蓝色,重新显示红色

七、CCScene的更多相关文章

  1. cocos2D(七)---- CCScene

    CCScene普通情况是游戏里面的根节点.称之为"场景",执行游戏时须要通过CCDirector启动第一个场景. 当然,游戏略微复杂一点的话.可能会包括非常多个场景,这就涉及到场景 ...

  2. 如何一步一步用DDD设计一个电商网站(七)—— 实现售价上下文

    阅读目录 前言 明确业务细节 建模 实现 结语 一.前言 上一篇我们已经确立的购买上下文和销售上下文的交互方式,传送门在此:http://www.cnblogs.com/Zachary-Fan/p/D ...

  3. CRL快速开发框架系列教程七(使用事务)

    本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...

  4. 《LoadRunner12七天速成宝典》来了

    看到自己的新书又要发行了,算算从09年第一本书开始,不知不觉已经是第四本书了(帮朋友合写的书不算),每次写完之后都会说太累了,不想再写了,但是却又次次反悔,吞下食言的苦果.如果非要说第四本书的感受,那 ...

  5. 【SAP业务模式】之ICS(七):IDOC配置

    这是ICS业务模式系列的最后一篇了,主要讲解IDOC的配置. 一.指定EDI传输的供应商逻辑地址 事务代码:WEL1 注意:上面逻辑地址是生产公司+内部客户.有以下两种情形: 1.如果内部客户都是纯数 ...

  6. 我的MYSQL学习心得(七) 查询

    我的MYSQL学习心得(七) 查询 我的MYSQL学习心得(一) 简单语法 我的MYSQL学习心得(二) 数据类型宽度 我的MYSQL学习心得(三) 查看字段长度 我的MYSQL学习心得(四) 数据类 ...

  7. Nodejs之MEAN栈开发(七)---- 用Angular创建单页应用(下)

    上一节我们走通了基本的SPA基础结构,这一节会更彻底的将后端的视图.路由.控制器全部移到前端.篇幅比较长,主要分页面改造.使用AngularUI两大部分以及一些优化路由.使用Angular的其他指令的 ...

  8. Go语言实战 - 网站性能优化第一弹“七牛云存储”

    由于用户纷纷反应山坡网的打开速度比较慢,所以两天前我们决定把服务器从linode迁移到阿里云. 整个迁移过程非常平滑,基本上一个小时就完成了.而且阿里云的配套设施提供的也很不错,运行状态监控什么的都有 ...

  9. redis成长之路——(七)

    扩展性封装 虽说现在StackExchange.Redis免费,万一到时候和servicestack.redis一样要收费呢,所以先留一口,后续的可以再处理 实例代码点击这里查看 redis成长之路- ...

随机推荐

  1. C#多态问题

    为什么对这个感觉趣呢.因为以前写过两篇关于这个多态和重载混合起来很乱的调用情况分析,自从哪以后,我自认为随便乱写一些继承多态的代码都应该难不到我.但是今天看到一段代码有一个地方计算错误了,所以有必要写 ...

  2. Excel中连接函数CONCATENATE()

    直接API: CONCATENATE 函数语法具有下列参数 (参数:为操作.事件.方法.属性.函数或过程提供信息的值.): Text1 必需. 要连接的第一个文本项. Text2, ... 可选. 其 ...

  3. PHP Memcached应用实现代码

    一.memcached 简介 在很多场合,我们都会听到 memcached 这个名字,但很多同学只是听过,并没有用过或实际了解过,只知道它是一个很不错的东东.这里简单介绍一下,memcached 是高 ...

  4. 重点关注之自定义序列化方式(Protobuf和Msgpack)

    除了默认的JSON和XML序列化器外,如果想使用其它格式的(比如二进制)序列化器,也是可以的.比如著名的Protobuf和Msgpack,它们都是二进制的序列化器,特点是速度快,体积小.使用方法如下. ...

  5. 【vmware vcp 5.1】安装及配置及笔记散记

    ESXi的几个命令技巧: ------------------------------------------------- alt-f1: 进入console alt-f2: 返回DCUI alt- ...

  6. static方法,属性,代码块初始化顺序和执行顺序

    http://greateryang.blog.163.com/blog/static/81953375201232621031508/

  7. POJ 2240 && ZOJ 1082 Arbitrage 最短路,c++ stl pass g++ tle 难度:0

    http://poj.org/problem?id=2240 用log化乘法为加法找正圈 c++ 110ms,g++tle #include <string> #include <m ...

  8. 二模 (11) day1

    第一题: 题目大意:用邻接矩阵给出一棵树(边权非负)上N个节点相互之间的最短路距离,求这棵树所有边权的和. 解题过程: 1.暂时还没想出来,待AC. 第二题: 题目大意:给出一些单词,然后建立Trie ...

  9. 分布式一致性原理—BASE

    定义 BASE是BasicallyAvailable(基本可用).Soft state(软状态)和Eventually consistent(最终一致性)三个短语的简写,是由来自eBay的架构师Dan ...

  10. 在OpenShift云空间上部署java web应用

    在使用之前先来介绍下OpenShift空间 OpenShift是由红帽推出逇一款面向开源开发人员开放的平台即服务(PaaS). OpenShift通过为开发人员提供在语言.框架和云上的更多的选择,使开 ...