离线下载
PDF版 ePub版

偶尔e网事 · 更新于 2018-11-28 11:00:43

游戏暂停和触摸屏蔽

一个游戏打到一半尿点咋整?难道要憋着。。。这不科学啊!

好吧,把暂停游戏和恢复游戏的功能加进去吧,否则也太对不起观众了!

1.暂停功能的加入

再给游戏加个层叫ControlLayer,这个层包含了2个元素,暂停功能和分数显示功能。分数显示和本地存储在后面介绍。

我们先看看暂停功能是怎么加入的。

    //加入暂停按钮
    bool ControlLayer::init()
    {
        bool bRet=false;
        do
        {
            CC_BREAK_IF(!CCLayer::init());

            CCSize winSize=CCDirector::sharedDirector()->getWinSize();

            //加入PauseMenu
            CCSprite* normalPause=CCSprite::create(CCSpriteFrameCache::sharedSpriteFrameCache()->spriteFrameByName("game_pause_nor.png"));
            CCSprite* pressedPause=CCSprite::create(CCSpriteFrameCache::sharedSpriteFrameCache()->spriteFrameByName("game_pause_pressed.png"));
            pPauseItem=CCMenuItemImage::create();//创建CCMenuItem
            pPauseItem->initWithNormalSprite(normalPause,pressedPause,NULL,this,menu_selector(ControlLayer::menuPauseCallback));//载入双态图和回调函数
            pPauseItem->setPosition(ccp(normalPause->getContentSize().width/2+10,winSize.height-normalPause->getContentSize().height/2-10));
            CCMenu *menuPause=CCMenu::create(pPauseItem,NULL);//创建CCMenu,可以这么理解CCMenuItem是CCMenu的孩子
            menuPause->setPosition(CCPointZero);
            this->addChild(menuPause,101);

            bRet=true;
        } while (0);

        return bRet;
    }
    //暂停按键的回调函数
    void ControlLayer::menuPauseCallback(CCObject* pSender)
    {
        if(!CCDirector::sharedDirector()->isPaused())//如果游戏处于正常状态
        {
            //更改为恢复按钮的双态
            pPauseItem->setNormalSpriteFrame(CCSpriteFrameCache::sharedSpriteFrameCache()->spriteFrameByName("game_resume_nor.png"));
            pPauseItem->setSelectedSpriteFrame(CCSpriteFrameCache::sharedSpriteFrameCache()->spriteFrameByName("game_resume_pressed.png"));
            CCDirector::sharedDirector()->pause();//暂停游戏,这是导演控制的
        }
        else//否则
        {
            //更改为暂停按钮的双态
            pPauseItem->setNormalSpriteFrame(CCSpriteFrameCache::sharedSpriteFrameCache()->spriteFrameByName("game_pause_nor.png"));
            pPauseItem->setSelectedSpriteFrame(CCSpriteFrameCache::sharedSpriteFrameCache()->spriteFrameByName("game_pause_pressed.png"));
            CCDirector::sharedDirector()->resume();//开麦拉!
        }
    }

这里没有使用CCMenuItemToggle的原因是它没办法实现一个图案的两个状态(normal和pressed),所以这里我手动进行替换。

2.游戏暂停状态下的触摸问题

如果就这么完了,那也弱爆了。。。游戏的调试过程中发现了这么一个问题,当游戏暂停的时候,主角飞机仍然可以跟随触摸移动,这个bug就坑爹了,你可以在快挂掉的时候按下pause,把飞机挪到安全的位置,然后再按下resume,死不了了。。。

原来cocos2d-x在暂停CCScene之后触摸仍然是有效的,所以我们需要在暂停之后屏蔽触摸。

这个的解决方案主要是两种:

(1)使用CCLayer的setTouchEnabled方法,但是这样可能会引起程序的崩溃,因为系统在派发触摸事件时发现响应列表为空,会触发一个断言。

(2)写一个NoTouchLayer,在这个层里响应触摸并吞噬触摸操作,使比它游戏级低的无法接收到触摸分发。但是优先级又不能高于CCMenu,也就是-128,不然恢复按钮也会被屏蔽,导致游戏无法恢复,除非是同一优先级。使用方法就是addChild和removeChild。关于触摸事件和优先级,请移步:,再次强调,触摸优先级和addChild的Z轴顺序无关。

    //NoTouchLayer.h
    class NoTouchLayer :
        public CCLayer
    {
    public:
        virtual bool init();

        // implement the "static node()" method manually
        LAYER_CREATE_FUNC(NoTouchLayer);

        virtual void registerWithTouchDispatcher();

        virtual bool ccTouchBegan (cocos2d::CCTouch *pTouch, cocos2d::CCEvent *pEvent);
        virtual void ccTouchMoved (cocos2d::CCTouch *pTouch, cocos2d::CCEvent *pEvent);
        virtual void ccTouchEnded (cocos2d::CCTouch *pTouch, cocos2d::CCEvent *pEvent);
    };

    //NoTouchLayer.cpp
    bool NoTouchLayer::init(){
        if (!CCLayer::init() )
        {
            return false;
        }
        setTouchEnabled(true);//设置触摸有效
        return true;
    }

    void NoTouchLayer::registerWithTouchDispatcher()
    {
        CCDirector::sharedDirector()->getTouchDispatcher()->addTargetedDelegate(this, -127 , true);//优先级低于-128(CCMenu),同时高于其他层(0),true表示吞噬触摸
        CCLayer::registerWithTouchDispatcher();
    }

    bool NoTouchLayer::ccTouchBegan (CCTouch *pTouch, CCEvent *pEvent)
    {
        return true;//返回true接收触摸
    }

    void NoTouchLayer::ccTouchMoved (CCTouch *pTouch, CCEvent *pEvent)
    {
    }

    void NoTouchLayer::ccTouchEnded (CCTouch *pTouch, CCEvent *pEvent)
    {
    }

3.第三种暂停屏蔽触摸的方法

因为游戏只有主角可以被触摸移动,所以只在PlaneLayer中的MoveTo函数里,做如下判断:

    if(isAlive && !CCDirector::sharedDirector()->isPaused())

这样就够了。如果游戏暂停就不让飞机移动。好像也没有什么问题。

效果图(暂停状态)