从零开始のcocos2dx生活(二)Node

节点 Node

文章目录

  • 节点 Node
    • 前言
    • 变量初始化
    • 创建一个节点对象
    • 获取节点依赖的计数器
    • 获取节点的描述(获取节点的Tag)
    • 节点的局部层顺序值(LocalZOrder)
      • 设置节点的LocalZOrder的值
      • 获取节点的LocalZOrder的值
      • 相同等级子节点具有相同ZOrder时设置绘制顺序
    • 节点的全局层顺序值(GlobalZOrder)
      • 设置节点的GlobalZOrder值
      • 获取节点的GlobalZOrder值
    • 节点的缩放(scale)
      • 设置缩放大小
      • 获取缩放大小
    • 节点位置(pisition)
      • 设置节点在父坐标系中的位置
      • 获取节点在父坐标系中的位置
      • 兼容旧版本
    • 节点倾斜(Skew)
      • 设置倾斜角
      • 获取倾斜角
    • 节点锚点(AnchorPoint)
      • 设置锚点
      • 获取锚点
      • 设置节点的锚点是否为(0,0)
      • 获取节点的锚点是否是(0,0)
    • 节点的原始大小(ContentSize)
      • 设置节点的原始大小
      • 获取节点的原始大小
    • 节点可见性(Visible)
      • 设置节点可见性
      • 确定节点是否可见
      • 设置节点的旋转角度
      • 获取节点的旋转角度
    • 节点的3d旋转(Rotation3D)
      • 设置节点的3d旋转
      • 获取节点的3d旋转
    • 节点的四元数(Quat)
      • 设置节点的四元数
      • 获取节点的四元数
    • 节点的旋转倾斜(RotationSkew)
      • 设置旋转倾斜
      • 获取旋转倾斜
      • 兼容旧版本
    • 子节点(Child)
      • 添加子节点
      • 获取子节点
      • 添加父节点
      • 获取父节点
    • 移除(remove)
      • 排序(Sort) !
    • 数据和用户标签
      • 设置
      • 获取
    • OPenGL
      • 获取OPenGL程序的状态
      • 设置OPenGL程序的状态
    • isRunning()
    • onEnter()
    • onExit()
    • cleanup()

前言

建议看完后面章节的知识后,多多回顾节点的代码,有助于加深理解。

节点是场景图中的基本元素。
最常见的节点对象有:场景、图层、精灵、菜单、标签、按钮等。

节点的主要特征是:
-它们可以包含其他节点对象(’ addChild ‘、’ removeChild ‘等)
-他们可以安排定期回调(’ schedule ', ’ unschedule ‘等)
-他们可以执行动作(’ runAction ', ’ stopAction '等)

子类化一个节点通常意味着(一个/所有):
-覆盖init来初始化资源和调度回调
-创建回调函数来处理时间的推移
-覆盖“绘制”以呈现节点

节点的属性:
-位置(默认:x=0, y=0)
-缩放(默认值:x=1, y=1)
-旋转(角度,顺时针方向)(默认为0)
-锚点(默认:x=0, y=0)
-contentSize(默认:宽度=0,高度=0)
-可见性(默认为真)

变量初始化

static const int INVALID_TAG = -1;  //设置标签初始值

enum {
        FLAGS_TRANSFORM_DIRTY = (1 << 0),  // 1
        FLAGS_CONTENT_SIZE_DIRTY = (1 << 1),  // 2
        FLAGS_RENDER_AS_3D = (1 << 3),  // 8

        FLAGS_DIRTY_MASK = (FLAGS_TRANSFORM_DIRTY | FLAGS_CONTENT_SIZE_DIRTY), // 或运算
    };

static Node * create(); //初始化节点,并返回一个加入到自动内存释放池中的节点

static int getAttachedNodeCount(); //获取附加在节点上的节点个数,(统计父节点上的子节点个数)

创建一个节点对象

Node * Node::create()
{
    Node * ret = new (std::nothrow) Node(); //创建一个名为ret的Node类型的对象
    if (ret && ret->init())									//判断ret是否创建成功并初始化对象ret
    {
        ret->autorelease();									//将对象ret加入内存自动回收池
    }
    else
    {
        CC_SAFE_DELETE(ret);								//如果创建对象失败,删除这个对象
    }
    return ret;															//返回创建成功的节点对象ret
}

获取节点依赖的计数器

int Node::getAttachedNodeCount()
{
    return __attachedNodeCount;  //返回被依赖的计数
}

获取节点的描述(获取节点的Tag)

std::string Node::getDescription() const
{
    return StringUtils::format(", _tag); //返回节点的Tag 
  													// _tag(Node::INVALID_TAG) 实际返回的就是INVALID_TAG
}

初始化器结束


节点的局部层顺序值(LocalZOrder)

LocalZOrder是用于对节点相对于其兄弟节点进行排序的“键”。

节点的父节点将根据LocalZOrder值对所有子节点排序。
如果两个节点具有相同的LocalZOrder,那么首先添加到子数组中的节点将位于数组中另一个节点的前面。

此外,场景图使用中序遍历,先遍历左子树,然后根节点,最后是右子树。
LocalZOrder值< 0的节点是左子树
而LocalZOrder >=0的节点是右子树。

设置节点的LocalZOrder的值

void Node::setLocalZOrder(std::int32_t z)
{
    if (getLocalZOrder() == z) //如果设置前后值相等,不做操作直接返回
        return;
    
    _setLocalZOrder(z); //否则初始化_setLocalZOrder变量
    if (_parent) //如果存在父节点
    {
        _parent->reorderChild(this, z);  //对父节点的子节点顺序重新排序
    }
    _eventDispatcher->setDirtyForNode(this); //将这个节点加入到事件调度中(加入节点监听器中,并将子节点也做相同操作)
}
//兼容旧版本
CC_DEPRECATED_ATTRIBUTE virtual void setZOrder(std::int32_t localZOrder) { setLocalZOrder(localZOrder); }

获取节点的LocalZOrder的值

virtual std::int32_t getLocalZOrder() const { return _localZOrder; }
//兼容旧版本
CC_DEPRECATED_ATTRIBUTE virtual std::int32_t getZOrder() const { return getLocalZOrder(); }

相同等级子节点具有相同ZOrder时设置绘制顺序

void updateOrderOfArrival();

节点的全局层顺序值(GlobalZOrder)

设置节点的GlobalZOrder值

定义节点渲染的顺序。
globalZOrder较低的节点首先渲染。

如果两个或多个节点具有相同的globalZOrder,则不保证渲染顺序。
当节点的globalZOrder == 0时。在这种情况下,使用场景图顺序。

默认情况下,所有节点的globalZOrder为0。这意味着默认情况下,一定会使用场景图顺序用于呈现节点。

当需要以 与场景图顺序不同的顺序 呈现节点时,GlobalZOrder非常有用。

void Node::setGlobalZOrder(float globalZOrder)
{
    if (_globalZOrder != globalZOrder) //如果设置前后不相等
    {
        _globalZOrder = globalZOrder;  //给_globalZOrder赋值
        _eventDispatcher->setDirtyForNode(this);  //将这个节点加入到事件调度中(加入节点监听器中,并将子节点也做相同操作)
    }
}

获取节点的GlobalZOrder值

virtual float getGlobalZOrder() const { return _globalZOrder; }

节点的缩放(scale)

设置缩放大小

//设置一个比例因子,用于乘以节点及其 子节点 的宽度。
void Node::setScaleX(float scaleX)
{
    if (_scaleX == scaleX)
        return;
    
    _scaleX = scaleX;
    _transformUpdated = _transformDirty = _inverseDirty = true; 
}
//设置一个比例因子,用于乘以节点及其 子节点 的高度。
void Node::setScaleY(float scaleY)
{
    if (_scaleY == scaleY)
        return;
    
    _scaleY = scaleY;
    _transformUpdated = _transformDirty = _inverseDirty = true;
}
//设置一个比例因子,用于乘以节点及其 子节点 的高度。
void Node::setScaleZ(float scaleZ)
{
    if (_scaleZ == scaleZ)
        return;
    
    _scaleZ = scaleZ;
    _transformUpdated = _transformDirty = _inverseDirty = true;
}
//设置一个比例因子,用于乘以节点及其 子节点 的宽度、高度、深度。
void Node::setScale(float scale)
{
    if (_scaleX == scale && _scaleY == scale && _scaleZ == scale)
        return;
    
    _scaleX = _scaleY = _scaleZ = scale;
    _transformUpdated = _transformDirty = _inverseDirty = true;
}
//它是一个比例因子,乘以节点及其子节点的宽度和高度。
void Node::setScale(float scaleX,float scaleY)
{
    if (_scaleX == scaleX && _scaleY == scaleY)
        return;
    
    _scaleX = scaleX;
    _scaleY = scaleY;
    _transformUpdated = _transformDirty = _inverseDirty = true; //标记脏数据
}

获取缩放大小

float Node::getScaleX() const
{
    return _scaleX;
}
float Node::getScaleY() const
{
    return _scaleY;
}
float Node::getScaleZ() const
{
    return _scaleZ;
}

//获取节点的缩放因子,当X和Y具有相同的缩放因子时
float Node::getScale(void) const
{
    CCASSERT( _scaleX == _scaleY, "CCNode#scale. ScaleX != ScaleY. Don't know which one to return"); // ?
    return _scaleX;
}

节点位置(pisition)

设置节点在父坐标系中的位置

//设置x,y轴的位置
void Node::setPosition(const Vec2& position)
{
    setPosition(position.x, position.y);  //(x, y)
}
//使用0-1设置x,y轴的位置
void Node::setPositionNormalized(const Vec2& position)
{
    if (_normalizedPosition.equals(position)) //修改前后相等,不做修改直接返回
        return;

    _normalizedPosition = position;  //给_normalizedPosition赋值
    _usingNormalizedPosition = true;  //标记 使用规范化位置
    _normalizedPositionDirty = true;	//标记 使用规范化位置的Dirty  // ? 标记脏数据
    _transformUpdated = _transformDirty = _inverseDirty = true;  // ? 标记脏数据重新计算渲染坐标矩阵  //在上面的修改之后,渲染用的坐标系矩阵未经计算,造成值的不对应,在当调用getNodeToParentTransform() 重新计算渲染用的坐标系矩阵之后会使用修改之后的值渲染,渲染之后的值对应了,不存在脏数据了,会将_transformDirty再设为false
}
//直接使用x、y设置位置
virtual void setPosition(float x, float y);
virtual void getPosition(float* x, float* y) const;
virtual void  setPositionX(float x);
virtual float getPositionX(void) const;
virtual void  setPositionY(float y);
virtual float getPositionY(void) const;
//设置节点的三维属性
void Node::setPosition3D(const Vec3& position)
{
    setPositionZ(position.z);
    setPosition(position.x, position.y);
}
//设置三维的z坐标
virtual void setPositionZ(float positionZ)
{
    if (_positionZ == positionZ)
        return;
    
    _transformUpdated = _transformDirty = _inverseDirty = true;

    _positionZ = positionZ;
}

获取节点在父坐标系中的位置

//_position
const Vec2& Node::getPosition() const
{
    return _position;
}
//获取节点的规范化位置 0-1
const Vec2& Node::getPositionNormalized() const
{
    return _normalizedPosition;
}
//获取节点的三维属性
Vec3 Node::getPosition3D() const
{
    return Vec3(_position.x, _position.y, _positionZ);
}
//获取节点的z坐标
float Node::getPositionZ() const
{
    return _positionZ;
}

兼容旧版本

virtual void setNormalizedPosition(const Vec2 &position) { setPositionNormalized(position); }

virtual const Vec2& getNormalizedPosition() const { return getPositionNormalized(); }

CC_DEPRECATED_ATTRIBUTE virtual void setVertexZ(float vertexZ) { setPositionZ(vertexZ); }

CC_DEPRECATED_ATTRIBUTE virtual float getVertexZ() const { return getPositionZ(); }

节点倾斜(Skew)

设置倾斜角

void Node::setSkewX(float skewX)
{
    if (_skewX == skewX)
        return;
    
    _skewX = skewX;
    _transformUpdated = _transformDirty = _inverseDirty = true;
}
void Node::setSkewY(float skewY)
{
    if (_skewY == skewY)
        return;
    
    _skewY = skewY;
    _transformUpdated = _transformDirty = _inverseDirty = true;
}

获取倾斜角

float Node::getSkewX() const
{
    return _skewX;
}
float Node::getSkewY() const
{
    return _skewY;
}

节点锚点(AnchorPoint)

锚点是所有转换和定位操作发生的点。
它就像节点上的一个大头针,它被“连接”到它的父节点上。
锚点是标准化的,比如百分比。(0,0)表示左下角,(1,1)表示右上角。
但是也可以使用大于(1,1)和小于(0,0)的值。
默认锚点是(0,0),因此它从节点的左下角开始。

设置锚点

//以百分比设置锚点
void Node::setAnchorPoint(const Vec2& point)
{
    if (! point.equals(_anchorPoint))
    {
        _anchorPoint = point;
        _anchorPointInPoints.set(_contentSize.width * _anchorPoint.x, _contentSize.height * _anchorPoint.y);
        _transformUpdated = _transformDirty = _inverseDirty = true;
    }
}

获取锚点

//以百分比为单位返回锚点
const Vec2& Node::getAnchorPoint() const
{
    return _anchorPoint;
}
//以绝对像素点为单位返回锚点
const Vec2& Node::getAnchorPointInPoints() const
{
    return _anchorPointInPoints;
}

设置节点的锚点是否为(0,0)

//默认值为false,而在图层和场景中为true
void Node::setIgnoreAnchorPointForPosition(bool newValue)
{
    if (newValue != _ignoreAnchorPointForPosition) 
    {
        _ignoreAnchorPointForPosition = newValue;
        _transformUpdated = _transformDirty = _inverseDirty = true;
    }
}
//兼容旧版本
CC_DEPRECATED_ATTRIBUTE virtual void ignoreAnchorPointForPosition(bool ignore) { setIgnoreAnchorPointForPosition(ignore); }

获取节点的锚点是否是(0,0)

bool Node::isIgnoreAnchorPointForPosition() const
{
    return _ignoreAnchorPointForPosition;
}

节点的原始大小(ContentSize)

设置节点的原始大小

void Node::setContentSize(const Size & size)
{
    if (! size.equals(_contentSize))
    {
        _contentSize = size;

        _anchorPointInPoints.set(_contentSize.width * _anchorPoint.x, _contentSize.height * _anchorPoint.y);
        _transformUpdated = _transformDirty = _inverseDirty = _contentSizeDirty = true;
    }
}

获取节点的原始大小

const Size& Node::getContentSize() const
{
    return _contentSize;
}

节点可见性(Visible)

设置节点可见性

//默认为可见的
void Node::setVisible(bool visible)
{
    if(visible != _visible)
    {
        _visible = visible;
        if(_visible)
            _transformUpdated = _transformDirty = _inverseDirty = true;
    }
}

确定节点是否可见

//可见即返回true,不可见返回false
bool Node::isVisible() const
{
    return _visible;
}

##节点的旋转角度(Rotation)

设置节点的旋转角度

//设置节点的旋转角度,这里是直接旋转,和下面的RotationSkew不一样
void Node::setRotation(float rotation)
{
    if (_rotationZ_X == rotation)
        return;
    
    _rotationZ_X = _rotationZ_Y = rotation;
    _transformUpdated = _transformDirty = _inverseDirty = true;
    
    updateRotationQuat();    //?
}

获取节点的旋转角度

float Node::getRotation() const
{
    CCASSERT(_rotationZ_X == _rotationZ_Y, "CCNode#rotation. RotationX != RotationY. Don't know which one to return");
    return _rotationZ_X;
}

节点的3d旋转(Rotation3D)

设置节点的3d旋转

void Node::setRotation3D(const Vec3& rotation)
{
    if (_rotationX == rotation.x &&
        _rotationY == rotation.y &&
        _rotationZ_X == rotation.z)
        return;
    
    _transformUpdated = _transformDirty = _inverseDirty = true;

    _rotationX = rotation.x;
    _rotationY = rotation.y;

    // rotation Z is decomposed in 2 to simulate Skew for Flash animations
  	//模拟flash动画的倾斜
    _rotationZ_Y = _rotationZ_X = rotation.z;
    
    updateRotationQuat();
}

获取节点的3d旋转

Vec3 Node::getRotation3D() const
{
    // rotation Z is decomposed in 2 to simulate Skew for Flash animations
    CCASSERT(_rotationZ_X == _rotationZ_Y, "_rotationZ_X != _rotationZ_Y");

    return Vec3(_rotationX,_rotationY,_rotationZ_X);
}

节点的四元数(Quat)

对于四元数的学习可以看我的另外一篇博客
cocos2dx 四元数

设置节点的四元数

//按四元数设置旋转
void Node::setRotationQuat(const Quaternion& quat)
{
    _rotationQuat = quat;
    updateRotation3D();
    _transformUpdated = _transformDirty = _inverseDirty = true;
}

获取节点的四元数

Quaternion Node::getRotationQuat() const
{
    return _rotationQuat;
}

节点的旋转倾斜(RotationSkew)

设置旋转倾斜

//设置水平旋转倾斜,模拟Flash的歪斜功能
void Node::setRotationSkewX(float rotationX)
{
    if (_rotationZ_X == rotationX)
        return;
    
    _rotationZ_X = rotationX;
    _transformUpdated = _transformDirty = _inverseDirty = true;
    
    updateRotationQuat();
}
//设置垂直旋转倾斜的角度,模拟Flash的歪斜功能
void Node::setRotationSkewY(float rotationY)
{
    if (_rotationZ_Y == rotationY)
        return;
    
    _rotationZ_Y = rotationY;
    _transformUpdated = _transformDirty = _inverseDirty = true;
    
    updateRotationQuat();
}

获取旋转倾斜

float Node::getRotationSkewX() const
{
    return _rotationZ_X;
}
float getRotationSkewY() const
{
    return _rotationZ_Y;
}

兼容旧版本

CC_DEPRECATED_ATTRIBUTE virtual void setRotationX(float rotationX) { return setRotationSkewX(rotationX); }

CC_DEPRECATED_ATTRIBUTE virtual void setRotationY(float rotationY) { return setRotationSkewY(rotationY); }

C_DEPRECATED_ATTRIBUTE virtual float getRotationY() const { return getRotationSkewY(); }

子节点(Child)

添加子节点

void Node::addChild(Node *child)
{
    CCASSERT( child != nullptr, "Argument must be non-nil"); //判断节点是否是空节点
    this->addChild(child, child->getLocalZOrder(), child->_name);//
}
//添加子节点同时设置localZOrder属性
void Node::addChild(Node *child, int zOrder)
{
    CCASSERT( child != nullptr, "Argument must be non-nil");
    this->addChild(child, zOrder, child->_name);
}
//添加子节点同时设置localZOrder和tag属性
void Node::addChild(Node *child, int localZOrder, int tag)
{    
    CCASSERT( child != nullptr, "Argument must be non-nil");
    CCASSERT( child->_parent == nullptr, "child already added. It can't be added again");

    addChildHelper(child, localZOrder, tag, "", true);
}
//添加子节点同时设置localZOrder和name属性
void Node::addChild(Node* child, int localZOrder, const std::string &name)
{
    CCASSERT(child != nullptr, "Argument must be non-nil");
    CCASSERT(child->_parent == nullptr, "child already added. It can't be added again");
    
    addChildHelper(child, localZOrder, INVALID_TAG, name, false);
}
//帮助添加子节点
void Node::addChildHelper(Node* child, int localZOrder, int tag, const std::string &name, bool setTag)
{
    auto assertNotSelfChild
        ( [ this, child ]() -> bool
          {
              for ( Node* parent( getParent() ); parent != nullptr;
                    parent = parent->getParent() )
                  if ( parent == child )
                      return false;
              
              return true;
          } );
    (void)assertNotSelfChild;
    
    CCASSERT( assertNotSelfChild(),
              "A node cannot be the child of his own children" );
    
    if (_children.empty())
    {
        this->childrenAlloc();
    }
    
    this->insertChild(child, localZOrder);
    
    if (setTag)
        child->setTag(tag);
    else
        child->setName(name);
    
    child->setParent(this);

    child->updateOrderOfArrival();

    if( _running )
    {
        child->onEnter(); //调用onEnter 增加依附计数 完成过渡
        // prevent onEnterTransitionDidFinish to be called twice when a node is added in onEnter
        if (_isTransitionFinished)
        {
            child->onEnterTransitionDidFinish(); //结束过渡
        }
    }
    
    if (_cascadeColorEnabled)
    {
        updateCascadeColor();
    }
    
    if (_cascadeOpacityEnabled)
    {
        updateCascadeOpacity();
    }
}

获取子节点

//遍历子节点,通过tag找到所需的节点
Node* Node::getChildByTag(int tag) const
{
    CCASSERT(tag != Node::INVALID_TAG, "Invalid tag");

    for (const auto child : _children)
    {
        if(child && child->_tag == tag)
            return child;
    }
    return nullptr;
}
//返回具有给定名称的节点,该节点可以转换为类型T
template <typename T>
    T getChildByTag(int tag) const { return static_cast<T>(getChildByTag(tag)); }
//通过name找到所需的节点
Node* Node::getChildByName(const std::string& name) const
{
    CCASSERT(!name.empty(), "Invalid name"); //需要有名字
    
    std::hash<std::string> h;
    size_t hash = h(name);
    
    for (const auto& child : _children)
    {
        // Different strings may have the same hash code, but can use it to compare first for speed
        if(child->_hashOfName == hash && child->_name.compare(name) == 0)
            return child;
    }
    return nullptr;
}
//返回具有给定名称的节点,该节点可以转换为类型T
 template <typename T>
    T getChildByName(const std::string& name) const { return static_cast<T>(getChildByName(name)); }
//查找节点
//  '//':只能放在搜索字符串的开头。这表明它将递归搜索。
//  '..':搜索应该向上移动到节点的父节点。只能放在字符串的末尾。
//  '/' :当搜索字符串放置在除开始位置以外的任何位置时,这表示搜索应该移动到节点的子节点。
//例如:
//enumerateChildren("//MyName", ...)  // 递归查找子节点
//enumerateChildren("[[:alnum:]]+", ...) // 匹配子节点的每个节点
//enumerateChildren("A[[:digit:]]", ...) // 查找A0-A9
//enumerateChildren("Abby/Normal", ...)  // 搜索并返回名为Normal父节点为Abby的任何节点
//enumerateChildren("//Abby/Normal", ...) // 递归搜索并返回名为Normal父节点为Abby的任何节点
void Node::enumerateChildren(const std::string &name, std::function<bool (Node *)> callback) const
{
    CCASSERT(!name.empty(), "Invalid name");
    CCASSERT(callback != nullptr, "Invalid callback function");
    
    size_t length = name.length();
    
    size_t subStrStartPos = 0;  // sub string start index
    size_t subStrlength = length; // sub string length
    
    // Starts with '//'?
    bool searchRecursively = false;
    if (length > 2 && name[0] == '/' && name[1] == '/')
    {
        searchRecursively = true;
        subStrStartPos = 2;
        subStrlength -= 2;
    }
    
    // End with '/..'?
    bool searchFromParent = false;
    if (length > 3 &&
        name[length-3] == '/' &&
        name[length-2] == '.' &&
        name[length-1] == '.')
    {
        searchFromParent = true;
        subStrlength -= 3;
    }
    
    // Remove '//', '/..' if exist
    std::string newName = name.substr(subStrStartPos, subStrlength);

    if (searchFromParent)
    {
        newName.insert(0, "[[:alnum:]]+/");
    }
    
    
    if (searchRecursively)
    {
        // name is '//xxx'
        doEnumerateRecursive(this, newName, callback);
    }
    else
    {
        // name is xxx
        doEnumerate(newName, callback);
    }
}
//返回节点的子节点数组
virtual Vector<Node*>& getChildren() { return _children; }
virtual const Vector<Node*>& getChildren() const { return _children; }
//获取子节点的个数
ssize_t Node::getChildrenCount() const
{
    return _children.size();
}

添加父节点

void Node::setParent(Node * parent)
{
    _parent = parent;
    _transformUpdated = _transformDirty = _inverseDirty = true;
}

获取父节点

virtual Node* getParent() { return _parent; }
virtual const Node* getParent() const { return _parent; }

移除(remove)

//将这个节点从父节点中移除,如果没有父节点则不操作
void Node::removeFromParent()
{
    this->removeFromParentAndCleanup(true);
}
void Node::removeFromParentAndCleanup(bool cleanup)
{
    if (_parent != nullptr)
    {
        _parent->removeChild(this,cleanup);
    } 
}
void ParallaxNode::removeChild(Node* child, bool cleanup)
{
    for( int i=0;i < _parallaxArray->num;i++)
    {
        PointObject *point = (PointObject*)_parallaxArray->arr[i];
        if (point->getChild() == child)
        {
            ccArrayRemoveObjectAtIndex(_parallaxArray, i, true);
            break;
        }
    }
    Node::removeChild(child, cleanup);
}
//移除子节点
//如果所有正在运行的动作和回调函数都在子节点上,cleanup为true,否则为false
void Node::removeChild(Node* child, bool cleanup /* = true */)
{
    if (_children.empty())
    {
        return;
    }

    ssize_t index = _children.getIndex(child);
    if( index != CC_INVALID_INDEX )
        this->detachChild( child, index, cleanup );
}
//通过tag移除子节点
void Node::removeChildByTag(int tag, bool cleanup/* = true */)
{
    CCASSERT( tag != Node::INVALID_TAG, "Invalid tag");

    Node *child = this->getChildByTag(tag);

    if (child == nullptr)
    {
        CCLOG("cocos2d: removeChildByTag(tag = %d): child not found!", tag);
    }
    else
    {
        this->removeChild(child, cleanup);
    }
}
//通过Name移除子节点
void Node::removeChildByName(const std::string &name, bool cleanup)
{
    CCASSERT(!name.empty(), "Invalid name");
    
    Node *child = this->getChildByName(name);
    
    if (child == nullptr)
    {
        CCLOG("cocos2d: removeChildByName(name = %s): child not found!", name.c_str());
    }
    else
    {
        this->removeChild(child, cleanup);
    }
}
//移除所有子节点
void Node::removeAllChildren()
{
    this->removeAllChildrenWithCleanup(true);
}
//删除所有子节点,并根据cleanup参数对所有正在运行的操作执行清理(cleanup)
void Node::removeAllChildrenWithCleanup(bool cleanup)
{
    // not using detachChild improves speed here
    for (const auto& child : _children)
    {
        // IMPORTANT:
        //  -1st do onExit
        //  -2nd cleanup
        if(_running)
        {
            child->onExitTransitionDidStart();
            child->onExit();
        }

        if (cleanup)
        {
            child->cleanup();
        }
#if CC_ENABLE_GC_FOR_NATIVE_OBJECTS
        auto sEngine = ScriptEngineManager::getInstance()->getScriptEngine();
        if (sEngine)
        {
            sEngine->releaseScriptObject(this, child);
        }
#endif // CC_ENABLE_GC_FOR_NATIVE_OBJECTS
        // set parent nil at the end
        child->setParent(nullptr);
    }
    
    _children.clear();
}

排序(Sort) !

void Node::reorderChild(Node *child, int zOrder)
{
    CCASSERT( child != nullptr, "Child must be non-nil");
    _reorderChildDirty = true;
    child->updateOrderOfArrival();
    child->_setLocalZOrder(zOrder);
}
//对所有的子节点排序,不需要手动调用,只会在绘制之前执行一次,以提高性能
void Node::sortAllChildren()
{
    if (_reorderChildDirty)
    {
        sortNodes(_children);
        _reorderChildDirty = false;
        _eventDispatcher->setDirtyForNode(this);
    }
}
//    
template<typename _T> inline
    static void sortNodes(cocos2d::Vector<_T*>& nodes)
    {
        static_assert(std::is_base_of<Node, _T>::value, "Node::sortNodes: Only accept derived of Node!");
#if CC_64BITS
        std::sort(std::begin(nodes), std::end(nodes), [](_T* n1, _T* n2) {
            return (n1->_localZOrder$Arrival < n2->_localZOrder$Arrival);
        });
#else
        std::sort(std::begin(nodes), std::end(nodes), [](_T* n1, _T* n2) {
            return (n1->_localZOrder == n2->_localZOrder && n1->_orderOfArrival < n2->_orderOfArrival) || n1->_localZOrder < n2->_localZOrder;
        });
#endif
    }

数据和用户标签

设置

UserData基本不用,一般使用UserDefault来记录数据,生成类似xml的文件。

    virtual void setTag(int tag);
    virtual void setName(const std::string& name);
    virtual void setUserData(void *userData);//可以看到这里的UserData是void类型的,也就是说可以记录任意数据
    virtual void setUserObject(Ref *userObject);  

获取

    virtual int getTag() const;
    virtual const std::string& getName() const;
    virtual void* getUserData() { return _userData; }
    virtual const void* getUserData() const { return _userData; }
    virtual Ref* getUserObject() { return _userObject; }
    virtual const Ref* getUserObject() const { return _userObject; }
    virtual int getTag() const;
    virtual const std::string& getName() const;
    virtual void* getUserData() { return _userData; }
    virtual const void* getUserData() const { return _userData; }
    virtual Ref* getUserObject() { return _userObject; }
    virtual const Ref* getUserObject() const { return _userObject; }

OPenGL

获取OPenGL程序的状态

//每个node私有的状态
GLProgram * Node::getGLProgram() const
{
    return _glProgramState ? _glProgramState->getGLProgram() : nullptr;
}
//兼容旧版本
CC_DEPRECATED_ATTRIBUTE GLProgram* getShaderProgram() const { return getGLProgram(); }
//通用的状态
GLProgramState* Node::getGLProgramState() const
{
    return _glProgramState;
}

设置OPenGL程序的状态

//设置私有的程序状态
void Node::setGLProgram(GLProgram* glProgram)
{
    if (_glProgramState == nullptr || (_glProgramState && _glProgramState->getGLProgram() != glProgram))
    {
        CC_SAFE_RELEASE(_glProgramState);
        _glProgramState = GLProgramState::getOrCreateWithGLProgram(glProgram);
        _glProgramState->retain();

        _glProgramState->setNodeBinding(this);
    }
}
//兼容旧版本
CC_DEPRECATED_ATTRIBUTE void setShaderProgram(GLProgram *glprogram) { setGLProgram(glprogram); }
//设置通用的程序状态
void Node::setGLProgramState(cocos2d::GLProgramState* glProgramState)
{
    if (glProgramState != _glProgramState)
    {
        CC_SAFE_RELEASE(_glProgramState);
        _glProgramState = glProgramState;
        CC_SAFE_RETAIN(_glProgramState);

        if (_glProgramState)
            _glProgramState->setNodeBinding(this);
    }
}

isRunning()

//返回节点是否正在运行
//如果节点正在运行,它将接受onEnter()、onExit()、update()等事件回调
bool Node::isRunning() const
{
    return _running;
}

onEnter()

//过渡效果开始之前调用,预处理一些动作
void Node::onEnter()
{
    if (!_running) //没有运行说明还未依附,之后依附,给依附计数器+1
    {
        ++__attachedNodeCount;
    }
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeJavascript)
    {
        if (ScriptEngineManager::sendNodeEventToJS(this, kNodeOnEnter))
            return;
    }
#endif
    
    if (_onEnterCallback)
        _onEnterCallback(); //调用自己的回调函数

    if (_componentContainer && !_componentContainer->isEmpty())
    {
        _componentContainer->onEnter();
    }
    
    _isTransitionFinished = false;
    
    for( const auto &child: _children) //子节点也会进入场景
        child->onEnter();
    
    this->resume();
    
    _running = true; //已经依附
    
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeLua)
    {
        ScriptEngineManager::sendNodeEventToLua(this, kNodeOnEnter);
    }
#endif
}
//完全进入场景时调用,添加背景乐等动作
void Node::onEnterTransitionDidFinish()
{
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeJavascript)
    {
        if (ScriptEngineManager::sendNodeEventToJS(this, kNodeOnEnterTransitionDidFinish))
            return;
    }
#endif
    
    if (_onEnterTransitionDidFinishCallback)
        _onEnterTransitionDidFinishCallback(); //调用结束过渡的回调函数

    _isTransitionFinished = true;
    for( const auto &child: _children) //子节点同步
        child->onEnterTransitionDidFinish();
    
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeLua)
    {
        ScriptEngineManager::sendNodeEventToLua(this, kNodeOnEnterTransitionDidFinish);
    }
#endif
}

onExit()

/* if(_running)
        {
            child->onExitTransitionDidStart();
            child->onExit();
        }
*/ //belong to void Node::removeAllChildren()

//在退出场景之前调用
void Node::onExitTransitionDidStart()
{
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeJavascript)
    {
        if (ScriptEngineManager::sendNodeEventToJS(this, kNodeOnExitTransitionDidStart))
            return;
    }
#endif
    
    if (_onExitTransitionDidStartCallback)
        _onExitTransitionDidStartCallback(); //调用回调函数
    
    for( const auto &child: _children) //子节点同步
        child->onExitTransitionDidStart();
    
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeLua)
    {
        ScriptEngineManager::sendNodeEventToLua(this, kNodeOnExitTransitionDidStart);
    }
#endif
}


//退出场景时调用
void Node::onExit()
{
    if (_running)
    {
        --__attachedNodeCount; //依附计数器-1
    }
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeJavascript)
    {
        if (ScriptEngineManager::sendNodeEventToJS(this, kNodeOnExit))
            return;
    }
#endif
    
    if (_onExitCallback)
        _onExitCallback(); //调用自己的回调函数
    
    if (_componentContainer && !_componentContainer->isEmpty())
    {
        _componentContainer->onExit();
    }
    
    this->pause(); //暂停事务
    
    _running = false; //不在运行中
    
    for( const auto &child: _children) //子节点也会被退出
        child->onExit(); 
    
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeLua)
    {
        ScriptEngineManager::sendNodeEventToLua(this, kNodeOnExit);
    }
#endif
}

cleanup()

//停止所有的操作和调度器
void Node::cleanup()
{
#if CC_ENABLE_SCRIPT_BINDING
    if (_scriptType == kScriptTypeJavascript)
    {
        if (ScriptEngineManager::sendNodeEventToJS(this, kNodeOnCleanup))
            return;
    }
    else if (_scriptType == kScriptTypeLua)
    {
        ScriptEngineManager::sendNodeEventToLua(this, kNodeOnCleanup);
    }
#endif // #if CC_ENABLE_SCRIPT_BINDING
    
    // actions 
    this->stopAllActions();  
    // timers
    this->unscheduleAllCallbacks();  

    
    for( const auto &child: _children) //子节点同步
        child->cleanup();
}

补充:
1、GlobalZorder、LocalZorder和orderOfArrival的区别
//渲染的优先级是Globalzorder > LocalZorder > orderOfArrival,类似于权重
值越大的越迟渲染,因为节点是使用二叉树的中序遍历来渲染的。
会将所有Globalzorder < 0,Globalzorder > 0,Globalzorder > 0,的节点都放在数组中。
localZorder区别节点本身与兄弟节点的渲染顺序,当两个或多个节点的localZorder值相等时会比较orderOfArrival的值。
orderOfArrival的值不需要人为的特别设置,在给节点添加子节点时会自动的自增, orderOfArrival的值默认为0。

2、setIgnoreAnchorPointForPosition();//一般Node的锚点默认为(0.5, 0.5),而Layer的锚点则在左下角(0,0),这个函数用来给Layer使用,layer的_ignoreAnchorPointForPosition值默认为true,忽略了锚点设置,如果自己设置了锚点也不会生效getChildren()

3、getChildren()有const修饰的问题 如果节点被const修饰,则只能调用被const修饰的getChildren函数。

4、_componentContainer的问题 相当于级联的操作

你可能感兴趣的:(从零开始のcocos2dx生活(二)Node)