quick-cocos2dx 悬浮节点(NotificationNode)

cocos2dx 开发游戏时,有时某些节点不需要随着场景的切换而销毁。但cocos2dx的机制只允许同时只有一个运行的场景,如果你的所有节点都是依附于这个场景的,那场景的切换必然带来节点的销毁。

比如,我们有一个悬浮图标,用来设置音乐音量,无论哪个场景都需要有这个按钮。就可以使用NotificationNode。

我遇到的问题是,收到服务器来的一条消息,客户端做一个提示,同时场景做一个切换。这就势必产生问题:提示文字首先加入到了要被销毁的场景,很快,新场景产生,就场景销毁,这个过程很短暂,所以你根本无法看到文字提示,就觉得莫名其妙切换了场景。

目前解决方法是收到这个提示消息后,延迟3秒钟显示,以保证场景已经切换完成。但觉得这个方法很不高大上,既然我有这个需求,别人肯定也会有,那么cocos本身就已经会提供更好的解决方法。

 

于是仔细查看Scene的C++代码,最终在Director的drawScene方法中,发现了这个片段

 // draw the scene

    if (_runningScene)

    {

        _runningScene->visit(_renderer, Mat4::IDENTITY, false);

        _eventDispatcher->dispatchEvent(_eventAfterVisit);

    }



    // draw the notifications node

    if (_notificationNode)

    {

        _notificationNode->visit(_renderer, Mat4::IDENTITY, false);

    }
_notificationNode 是一个亮点,于是各种百度google之后,觉得这才是最王道的解决方法。

于是我在quick中是这样调用的,myapp.lua 的ctor 方法 增加 如下代码:
cc.Director:getInstance():setNotificationNode(require("app.views.NotificationLayer").new())
NotificationLayer 源码是这样的
--

-- Created by IntelliJ IDEA.

-- User: Elan

-- Date: 15-7-27 下午5:37

-- To change this template use File | Settings | File Templates.

--



local NotificationLayer = class("NotificationLayer", function()

    return display.newNode()

end)



function NotificationLayer:ctor()

    self:setNodeEventEnabled(true)

    self:addSprite()

    self:hide()

end



function NotificationLayer:registerNotificationCenter()

    self.showNotificationHandle = GameDataCenter:addEventListener("showNotification", handler(self, self.showAni))

end



function NotificationLayer:unregisterNotificationCenter()

    GameDataCenter:removeEventListener(self.showNotificationHandle)

end





function NotificationLayer:onEnter()

    self:registerNotificationCenter()

    print("onEnter..............")

end



function NotificationLayer:onExit()

    self:unregisterNotificationCenter()

    print("onExit..............")

end



function NotificationLayer:addSprite()

    self.bg = cc.ui.UIImage.new("#images/common/ui/shangfangtishi.png")

    :align(display.BOTTOM_CENTER, display.cx, display.top)

    :addTo(self)



    self.bgw = self.bg:getContentSize().width

    self.bgh = self.bg:getContentSize().height



    self.txtLabel = cc.ui.UILabel.new({

        text = "",

        size = 24,

        color = ccYELLOW,

        align = ui.TEXT_ALIGN_CENTER

    })

    :align(display.CENTER, self.bgw/2, self.bgh/2)

    :addTo(self.bg)

end



function NotificationLayer:showAni(data)

    self.bg:stopAllActions()

    self:show()

    local msg = data.Responsedata

    self.txtLabel:setString(msg)

    local action = transition.sequence({

        cc.MoveTo:create(0.5, cc.p(display.cx, display.top - self.bgh)),

        cc.DelayTime:create(2),

        cc.MoveBy:create(0.1, cc.p(display.cx, display.top)),

        cc.CallFunc:create(function()

            self:hide()

        end)

    })

    self.bg:runAction(action)

end



return NotificationLayer

 

我在其他地方触发 showNotification 事件后,发现背景板并没有顺利的移动出来,就是说我的action并没有执行成功。于是单步调试,发现runAction调用时,node本身的_running是false,所以action并没有执行。
看网上很多童鞋说再调用一下onEnter方法,确实是,只有onEnter方法里面_running才会被设置为true。于是我在lua里面调用onEnter,真是笨到家了。因为lua里面的onEnter是C++里面的onEnter的回调,所以啥作用都没起。

于是我在Director里面增加调用onEnter()方法,然后很成功的执行了action。
void Director::setNotificationNode(Node *node)

{

    CC_SAFE_RELEASE(_notificationNode);

    _notificationNode = node;

    node->onEnter(); //  add by Elan 2015.7.27

    CC_SAFE_RETAIN(_notificationNode);

}

 

但是,我关掉player的时候,又报错了:

Node still marked as running on node destruction! Was base class onExit() called in derived class onExit() implementations?

C++一看,node的析构函数报错,node 的_running 为true的时候销毁这个node就会报这个错。_running只在OnExit()的会被设为false。于是我在Director 的析构函数中又增加了一句。部分代码如下。

Director::~Director(void)

{

    CCLOGINFO("deallocing Director: %p", this);



    if (_notificationNode)

    {

        _notificationNode->onExit(); // add by Elan 2015.7.27

    }



    CC_SAFE_RELEASE(_FPSLabel);

    CC_SAFE_RELEASE(_drawnVerticesLabel);

    CC_SAFE_RELEASE(_drawnBatchesLabel);



    CC_SAFE_RELEASE(_runningScene);

    CC_SAFE_RELEASE(_notificationNode);

    CC_SAFE_RELEASE(_scheduler);

    CC_SAFE_RELEASE(_actionManager);



    CC_SAFE_RELEASE(_scriptEventCenter);

 

现在就完美啦。性能神马的会不会有什么影响,只能后续测试啦。

 

你可能感兴趣的:(notification)