cocos2dx3.3开发FlappyBird总结十三:数字特效类

简介:

由于显示得分其实是数字精灵的组合,因此需要先设计一个存储数字精灵数列的类:

#ifndef __EngryBird__NumberSeries__
#define __EngryBird__NumberSeries__

#include "cocos2d.h"

/**
 * This class is for ScoreNumber, and it will store a list of sprite frames.
 * With it, you can load number sprite with name format and get sprite frame
 * with index.
 */
class NumberSeries : public cocos2d::Ref {
public:
  /**
   * The default constructor
   */
  NumberSeries();

  /**
   * The default destructor
   */
  ~NumberSeries();

  /**
   * The init method, will init the super init method first
   *
   * @return true if succeeded, otherwise false
   */
  virtual bool init();

  CREATE_FUNC(NumberSeries);

  /**
   * Load sprite frame with a format name
   *
   * @param format The name format, eg. "number1_%d"
   * @param base The begin index
   */
  void loadNumber(const char *foramt, int base = 0);

  /**
   * Get sprite frame with an index
   *
   * @param index The index in the _numberVector
   *
   * @return a sprite frame object
   */
  cocos2d::SpriteFrame* at(int index);

private:
  /**
   * Store number sprite frames
   */
  cocos2d::Vector<cocos2d::SpriteFrame *> _numberVector;
};

#endif /* defined(__EngryBird__NumberSeries__) */

这个类只有一个成员变量,_numberVector,存储的是一组数字精灵帧,这样就可以通过数字作为下标,直接获取对应的精灵帧。

在初始化时:

bool NumberSeries::init() {
  _numberVector = cocos2d::Vector<SpriteFrame *>(kMaxNumberCount);

  return true;
}

因此数字就是0~9这几个,所以是固定的,写成一个宏常量

下面这个方法是获取得分组合精灵,如得分为42,则把对应的精灵帧4和2加入到组合中,

void NumberSeries::loadNumber(const char *foramt, int base) {
  for (int i = base; i < kMaxNumberCount + base; ++i) {
    char name[20];
    sprintf(name, foramt, i);

    auto frame = AtlasLoader::getInstance()->getSpriteFrame(name);
    _numberVector.pushBack(frame);
  }
}

base表示从什么数字开始,这样写是因为有的精灵的名称不是从0开始的,如bird_045~bird_088,那么base就是45了

获取数字精灵帧的时候,传数字过来,就是获取对应的数字精灵帧了,

SpriteFrame* NumberSeries::at(int index) {
  if (index >= 0 && index < _numberVector.size()) {
    return _numberVector.at(index);
  }

  return NULL;
}

考虑代码的健壮性,增加了范围的判断

下面是数字特效类了,设计为全局共享类,
首先,显示的方式为居中,居左,居右,因此先设计一个枚举来表示。

/**
 * The alignment
 */
typedef enum {
  kGravityDirectionCenter = 1,
  kGravityDirectionLeft,
  kGravityDirectionRight
} GravityDirection;

看看加载得分精灵方法:

bool ScoreNumber::loadNumber(const char *name, const char *format, int base) {
  auto series = NumberSeries::create();
  series->loadNumber(format, base);

  _numberContainer.insert(name, series);

  return true;
}

这里_numberContainer是一个Map类型的容器,用于存储得分,因为游戏不断重复的话,同样的得分,就可以直接获取,而不用重复了。

下面这个方法是把数字得分转换成精灵数字,并调整方向:

Node* ScoreNumber::convert(const char *name, int number, GravityDirection direction) {
// 首先根据key,取得数字组合
  auto series = _numberContainer.at(name);
  auto zero = Sprite::createWithSpriteFrame(series->at(0));
  // 如果得分为0,直接显示
  if (number == 0) {
    zero->setAnchorPoint(Vec2(0.5, 0));

    return zero;
  }

  // seperate the number and load number sprite into the node
  // 使用node节点来容纳数字精灵
  auto node = Node::create();
  float totalWidth = 0.0f;
  while (number) {
  // 循环获取每一位数字,然后获取对应的精灵,添加到node中并计算
  // node的总宽
    int tmp = number % 10;
    auto sprite = Sprite::createWithSpriteFrame(series->at(tmp));

    totalWidth += sprite->getContentSize().width;
    node->addChild(sprite);

    number /= 10;
  }

  // set the content size of node
  node->setContentSize(Size(totalWidth, zero->getContentSize().height));

  // caculate the width of each number
  float perWidth = totalWidth / node->getChildrenCount();
  ssize_t index = 0;
  float anchorX = 0;
  bool isMinus = true;

  if (direction == kGravityDirectionCenter) {
    anchorX = 0.5f;
    index = node->getChildrenCount() / 2;
  } else if (direction == kGravityDirectionRight) {
    anchorX = 1.0f;
    index = node->getChildrenCount();
  } else if (direction == kGravityDirectionLeft) {
    anchorX = 0.0f;
    isMinus = false;
    index = 0;
  }

  // 获取显示的方式来调整精灵的位置
  for (auto child : node->getChildren()) {
    child->setAnchorPoint(Vec2(anchorX, 0));

    float posX = perWidth * (isMinus ? index-- : index++);
    child->setPositionX(posX);
  }

  return node;
}

下一步,先说明一下用到的宏

目录
相关文章
|
人工智能 智能设计 算法
游戏编程之十三 3D类游戏
游戏编程之十三 3D类游戏
133 0
|
6月前
|
图形学
【用unity实现100个游戏之18】从零开始制作一个类CSGO/CS2、CF第一人称FPS射击游戏——基础篇3(附项目源码)
【用unity实现100个游戏之18】从零开始制作一个类CSGO/CS2、CF第一人称FPS射击游戏——基础篇3(附项目源码)
66 0
|
6月前
|
定位技术 图形学 开发者
【用unity实现100个游戏之18】从零开始制作一个类CSGO/CS2、CF第一人称FPS射击游戏——基础篇1(附项目源码)
【用unity实现100个游戏之18】从零开始制作一个类CSGO/CS2、CF第一人称FPS射击游戏——基础篇1(附项目源码)
91 0
|
6月前
|
图形学
【用unity实现100个游戏之18】从零开始制作一个类CSGO/CS2、CF第一人称FPS射击游戏——基础篇2(附项目源码)
【用unity实现100个游戏之18】从零开始制作一个类CSGO/CS2、CF第一人称FPS射击游戏——基础篇2(附项目源码)
78 0
|
6月前
|
图形学
【用unity实现100个游戏之17】从零开始制作一个类幸存者肉鸽(Roguelike)游戏5(附项目源码)(下)
【用unity实现100个游戏之17】从零开始制作一个类幸存者肉鸽(Roguelike)游戏5(附项目源码)
97 0
|
6月前
|
图形学
【用unity实现100个游戏之17】从零开始制作一个类幸存者肉鸽(Roguelike)游戏5(附项目源码)(上)
【用unity实现100个游戏之17】从零开始制作一个类幸存者肉鸽(Roguelike)游戏5(附项目源码)
79 0
|
6月前
【2023Unity游戏开发教程】零基础带你从小白到超神02——新建第一个项目
【2023Unity游戏开发教程】零基础带你从小白到超神02——新建第一个项目
28 0
|
C# 图形学
边学边做的第一个Unity小游戏
学Unity开发,肯定先要了解小Unity这个开发工具长什么样。 第一次看到Unity界面的时候,我第一反应是,我靠,这不是犀牛么。学工业设计的人应该接触过犀牛这款3D建模软件,主界面也是一个3D透视界面。
1540 0
|
Android开发
Android项目实战(四十一):游戏和视频类型应用 状态栏沉浸式效果
原文:Android项目实战(四十一):游戏和视频类型应用 状态栏沉浸式效果   需求:    手机app ,当打游戏或者全屏看视频的时候会发现这时候手机顶部的状态栏是不显示的,当我们从手机顶端向下进行滑动或手机底端向上滑动的时候,状态栏会显示出来,如果短暂的几秒时间没有操作的话,状态栏会再次隐藏。
1047 0