我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Cocos2D-X基本教程:详解Cocos2D-X 3.0中的坐标系

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Cocos2D-X基本教程:详解Cocos2D-X 3.0中的坐标系

  我们在游戏开发中经常要确定精灵或其他UI控件在屏幕的位置,应该怎么做呢?本篇教程将详细解读Cocos2d-x中的Cocos2D-X 3.0中的坐标系及其运用。

  笛卡尔坐标系

  笛卡尔坐标系中定义右手系原点在左下角,x向右,y向上,z向外,OpenGL坐标系为笛卡尔右手系。

Cocos2D-X基本教程:详解Cocos2D-X 3.0中的坐标系_Cocos2D-X坐标系_Cocos2D-X教程_Cocos2D-X基本_编程学习网

  屏幕坐标系和Cocos2d坐标系

  标准屏幕坐标系使用和OpenGL不同的坐标系,而Cocos2d则使用和OpenGL相同的坐标系。

  iOS, Android, Windows Phone等在开发应用时使用的是标准屏幕坐标系,原点为屏幕左上角,x向右,y向下。

  Cocos2d坐标系和OpenGL坐标系一样,原点为屏幕左下角,x向右,y向上。

  在开发中,我们经常会提到两个比较抽象的概念-世界坐标系和本地坐标系。这两个概念可以帮助我们更好的理解节点在Cocos2d坐标系中的位置以及对应关系。

  世界坐标系(World Coordinate) VS 本地坐标系(Node Local)

  世界坐标系也叫做绝对坐标系,是游戏开发中建立的概念。因此,“世界”指游戏世界。cocos2d中的元素是有父子关系的层级结构,我们通过Node的setPosition设定元素的位置使用的是相对与其父节点的本地坐标系而非世界坐标系。最后在绘制屏幕的时候cocos2d会把这些元素的本地坐标映射成世界坐标系坐标。

  本地坐标系也叫相对坐标系,是和节点相关联的坐标系。每个节点都有独立的坐标系,当节点移动或改变方向时,和该节点关联的坐标系将随之移动或改变方向。

  锚点(Anchor Point)

  将一个节点添加到父节点里面时,需要设置其在父节点上的位置,本质上是设置节点的锚点在父节点坐标系上的位置。

  Anchor Point的两个参数都在0~1之间。它们表示的并不是像素点,而是乘数因子。(0.5, 0.5)表示Anchor Point位于节点长度乘0.5和宽度乘0.5的地方,即节点的中心

  在Cocos2d-x中Layer的Anchor Point为默认值(0, 0),其他Node的默认值为(0.5, 0.5)。

  我们用以下代码为例,使用默认Anchor Point值,将红色层放在屏幕左下角,绿色层添加到红色层上:

  auto red = LayerColor::create(Color4B(255, 100, 100, 128), visibleSize.width/2, visibleSize.height/2);

  auto green = LayerColor::create(Color4B(100, 255, 100, 128), visibleSize.width/4, visibleSize.height/4);

  red->addChild(green);

  this->addChild(red, 0);

  我们用以下代码为例,将红色层的Anchor Point设为中点放在屏幕中央,绿色层添加到红色层上,绿色层锚点为右上角:

  注:因为Layer比较特殊,它默认忽略锚点,所以要调用ignoreAnchorPointForPosition()接口来改变锚点,关于ignoreAnchorPointForPosition()接口的使用说明,我们将在后面详细讲解。

  auto red = LayerColor::create(Color4B(255, 100, 100, 128), visibleSize.width/2, visibleSize.height/2);

  red->ignoreAnchorPointForPosition(false);

  red->setAnchorPoint(Point(0.5, 0.5));

  red->setPosition(Point(visibleSize.width/2 + origin.x, visibleSize.height/2 + origin.y));

  auto green = LayerColor::create(Color4B(100, 255, 100, 128), visibleSize.width/4, visibleSize.height/4);

  green->ignoreAnchorPointForPosition(false);

  green->setAnchorPoint(Point(1, 1));

  red->addChild(green);

  this->addChild(red, 0);

  忽略锚点(Ignore Anchor Point)

  Ignore Anchor Point全称是ignoreAnchorPointForPosition,作用是将锚点固定在一个地方。

  如果设置其值为true,则图片资源的Anchor Pont固定为左下角,否则即为所设置的位置。

  我们用以下代码为例,将两个层的ignoreAnchorPointForPosition设为true,并将绿色的层添加到红色的层上:

  auto red = LayerColor::create(Color4B(255, 100, 100, 128), visibleSize.width/2, visibleSize.height/2);

  red->ignoreAnchorPointForPosition(true);

  red->setPosition(Point(visibleSize.width/2 + origin.x, visibleSize.height/2 + origin.y));

  auto green = LayerColor::create(Color4B(100, 255, 100, 128), visibleSize.width/4, visibleSize.height/4);

  green->ignoreAnchorPointForPosition(true);

  red->addChild(green);

  this->addChild(red, 0);

  VertexZ,PositionZ和zOrder

  VerextZ是OpenGL坐标系中的Z值

  PositionZ是Cocos2d-x坐标系中Z值

  zOrder是Cocos2d-x本地坐标系中Z值

  在实际开发中我们只需关注zOrder。

  可以通过setPositionZ接口来设置PositionZ。

  以下是setPositionZ接口的说明:

  Sets the 'z' coordinate in the position. It is the OpenGL Z vertex value.

  即PositionZ的值即为opengl的z值VertexZ。同样节点的PositionZ也是决定了该节点的渲染顺序,值越大,但是与zOrder不同的区别在于,PositionZ是全局渲染顺序即在根节点上的渲染顺序,而zOrder则是局部渲染顺序,即该节点在其父节点上的渲染顺序,与Node的层级有关。用以下事例来说明:

  auto red = LayerColor::create(Color4B(255, 100, 100, 255), visibleSize.width/2, visibleSize.height/2);

  red->ignoreAnchorPointForPosition(false);

  red->setPosition(Point(visibleSize.width / 2, visibleSize.height / 2));

  auto green = LayerColor::create(Color4B(100, 255, 100, 255), visibleSize.width/4, visibleSize.height/4);

  green->ignoreAnchorPointForPosition(false);

  green->setPosition(Point(visibleSize.width / 2, visibleSize.height / 2 - 100));

  red->setPositionZ(1);

  green->setPositionZ(0);

  this->addChild(red, 0);

  this->addChild(green, 1);

  虽然green的zOrder大于red的zOder,但是因为red的PositionZ较大,所以red还是在green上面显示。

  触摸点(Touch position)

  所以在处理触摸事件时需要用重写以下四个函数:

  virtual bool onTouchBegan(Touch *touch, Event * event);

  virtual void onTouchEnded(Touch *touch, Event * event);

  virtual void onTouchCancelled(Touch *touch, Event * event);

  virtual void onTouchMoved(Touch *touch, Event * event);

  在函数中获取到touch,我们在设计游戏逻辑时需要用到触摸点在Cocos2d坐标系中的位置,就需要将touch的坐标转换成OpenGL坐标系中的点坐标。

  Touch position是屏幕坐标系中的点,OpenGL position是Cocos2d-x用到的OpenGL坐标系上的点坐标。通常我们在开发中会使用两个接口getLocation()和getLocationInView()来进行相应坐标转换工作。

  在开发中一般使用getLocation()获取触摸点的GL坐标,而getLocation()内部实现是通过调用Director::getInstance()->convertToGL(_point);返回GL坐标。

  此外,关于世界坐标系和本地坐标系的相互转换,在Node中定义了以下四个常用的坐标变换的相关方法。

  // 把世界坐标转换到当前节点的本地坐标系中

  Point convertToNodeSpace(const Point& worldPoint) const;

  // 把基于当前节点的本地坐标系下的坐标转换到世界坐标系中

  Point convertToWorldSpace(const Point& nodePoint) const;

  // 基于Anchor Point把基于当前节点的本地坐标系下的坐标转换到世界坐标系中

  Point convertToNodeSpaceAR(const Point& worldPoint) const;

  // 基于Anchor Point把世界坐标转换到当前节点的本地坐标系中

  Point convertToWorldSpaceAR(const Point& nodePoint) const;

  下面通过一个例子来说明这四个方法的理解和作用:

  auto *sprite1 = Sprite::create("HelloWorld.png");

  sprite1->setPosition(ccp(20,40));

  sprite1->setAnchorPoint(ccp(0,0));

  this->addChild(sprite1); //此时添加到的是世界坐标系,也就是OpenGL坐标系

  auto *sprite2 = Sprite::create("HelloWorld.png");

  sprite2->setPosition(ccp(-5,-20));

  sprite2->setAnchorPoint(ccp(1,1));

  this->addChild(sprite2); //此时添加到的是世界坐标系,也就是OpenGL坐标系

  //将 sprite2 这个节点的坐标ccp(-5,-20) 转换为 sprite1节点 下的本地(节点)坐标系统的 位置坐标

  Point point1 = sprite1->convertToNodeSpace(sprite2->getPosition());

  //将 sprite2 这个节点的坐标ccp(-5,-20) 转换为 sprite1节点 下的世界坐标系统的 位置坐标

  Point point2 = sprite1->convertToWorldSpace(sprite2->getPosition());

  log("position = (%f,%f)",point1.x,point1.y);

  log("position = (%f,%f)",point2.x,point2.y);

  运行结果:

  Cocos2d: position = (-25.000000,-60.000000)

  Cocos2d: position = (15.000000,20.000000)

  其中:Point point1 = sprite1->convertToNodeSpace(sprite2->getPosition());

  相当于sprite2这个节点添加到(实际没有添加,只是这样理解)sprite1这个节点上,那么就需要使用sprite1这个节点的节点坐标系统,这个节点的节点坐标系统的原点在(20,40),而sprite1的坐标是(-5,-20),那么经过变换之后,sprite1的坐标就是(-25,-60)。

  其中:Point point2 = sprite1->convertToWorldSpace(sprite2->getPosition());

  此时的变换是将sprite2的坐标转换到sprite1的世界坐标系下,而其中世界坐标系是没有变化的,始终都是和OpenGL等同,只不过sprite2在变换的时候将sprite1作为了”参照“而已。所以变换之后sprite2的坐标为:(15,20)。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Cocos2D-X基本教程:详解Cocos2D-X 3.0中的坐标系

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Cocos2D-X基本教程:详解Cocos2D-X 3.0中的坐标系

编程学习网:我们在游戏开发中经常要确定精灵或其他UI控件在屏幕的位置,应该怎么做呢?本篇教程将详细解读Cocos2D-X中的Cocos2D-X 3.0中的坐标系及其运用。
Cocos2D-X基本教程:详解Cocos2D-X 3.0中的坐标系

Cocos2D-X层基本教程:在Cocos2D-X中创建屏蔽层

编程学习网:在游戏开发中,有时我们会需要设置一个屏蔽层,我们的同学们是不是对此感到束手无策,不知如何设置呢?本篇教程将为你讲解如何在Cocos2D-X中创建屏蔽层。
Cocos2D-X层基本教程:在Cocos2D-X中创建屏蔽层

塔防游戏必备 Cocos2D-X中动画加减速基本教程

编程学习网:在塔防游戏中我们经常会看到动画减速与加速的应用,这些效果在游戏开发中要怎么制作呢?本篇教程将以塔防游戏为例讲解Cocos2D-X中动画加减速问题。
塔防游戏必备 Cocos2D-X中动画加减速基本教程

编程热搜

  • 教你如何为你的Cocos2D-X资源加密解密
    编程学习网:我们平时在用Cocos2D-X时都不想让别人看到自己的文件,本篇教程将教你如何为你的Cocos2D-X资源加密解密。
    教你如何为你的Cocos2D-X资源加密解密
  • 教你如何用Cocos2D实现3d效果
    编程学习网:经过了之前几篇教程,相信各位同学已经对Cocos2D不陌生了,本篇教程将教你如何用Cocos2D实现3d效果。
    教你如何用Cocos2D实现3d效果
  • 零基础学习之教你如何用Cocos2D实现画中画效果
    编程学习网:没学过Cocos2D?没关系,多看看我们的教程,一步步成为游戏开发大神。本篇教程将教你如何用Cocos2D实现画中画效果。
    零基础学习之教你如何用Cocos2D实现画中画效果
  • 解析安卓Cocos2D-X2转移项目到Cocos2D-X3过程
    编程学习网:相信有一部分同学都经历过将项目从Cocos2D-X2转到Cocos2D-X3的麻烦,本篇教程将解析将项目从Cocos2D-X2转到Cocos2D-X3的过程。
    解析安卓Cocos2D-X2转移项目到Cocos2D-X3过程
  • 游戏脚本设计基础教程
    编程学习网:类游戏编程,特别是RPG脚本起着驱动整个游戏进程的作用。事件的运作建立在脚本的基础上,而脚本的设计建立在引擎的基础上,所以设计脚本之前因该想一想引擎,好的脚本对剧情的描述具有简单、准确的性质。
    游戏脚本设计基础教程
  • 成为游戏设计师必须具备的条件
    编程学习网:很多人喜欢玩游戏,他们总在不停感叹:这款游戏实在太精彩了,我要是能设计出这样的游戏就!而在他们行动之前,还有许多需要解答的问题罗列在前,我要如何成为游戏设计师?我需要具备哪些技能和素质?游戏设计师究竟在做什么工作?那么,游戏设计师有哪些任务呢?我们将讨论这个问题。
    成为游戏设计师必须具备的条件
  • 贪食蛇小游戏开发设计基础教程
    编程学习网: 贪吃蛇是家喻户晓的益智类小游戏,大家小时候应该都有玩过,编程学习网这里就不多介绍了,本教程将教你用MicrosoftVisualC++来制作它。
    贪食蛇小游戏开发设计基础教程
  • 成为优秀的Cocos2D程序员需要的十个品质
    编程学习网:相信各位同学都在为了成为优秀的Cocos2D程序员而奋斗着,那么一个优秀的Cocos2D程序员需要具备什么品质呢?本篇教程为你揭秘成为一个优秀的Cocos2D程序员所需要的十个品质。
    成为优秀的Cocos2D程序员需要的十个品质
  • Unity3d脚本基础
    最好用游戏引擎开发游戏,推荐Unity3D引擎,该引擎学习更简单,更易上手。游戏引擎可以编辑你的游戏场景、角色和游戏需要的东西。还有,学习编写脚本。编写脚本实际上就是编程。例如,你要在游戏中按键盘方向键来控制角色行走,这必须通过编写脚本。因此,脚本是游戏的逻辑。小编推荐你去学C编程,Unity3D通常是用C语言编写的。
    Unity3d脚本基础
  • 实例教程解析制作flash小游戏
    编程学习网:flash小游戏在游戏开发中是属于比较简单的类型,本例为FlashAS3.0实例教程,介绍射击类游戏的制作,主要分游戏界面的制作和类的编写两部分,从简单的开始做起,跟着教程动手做做看吧。
    实例教程解析制作flash小游戏

目录