CCTableView 与 CCMenu巧妙结合 . 分类: cocos2d/cocos2s-x2012-12-13 17:524740人阅读评论(7)收藏举报今天被赋予了一项新的任务,那就是看看cocos2d新版本的CCTableView能不能解决项目TableView的Bug。 项目内的TableView会存在如下bug: •当TableView里面的MenuItem滚动出View方框时,用户应该不能在View方框外选中该MenuItem的。但用户却能点击到。•当TableView上面还有一层Layer时,当点击Layer的时候,TableView里面的MenuITem也会被选中。 第一个bug的主要原因是因为CCMenu的优先级太高(-128),另外一点就是即使当CCMenu优先级低了,也很难让二者行为与一般的窗口空间相同,因为他们之间没有正常的窗口父子关系。cocos2d的消息处理是基于优先级,也就是说即使被覆盖的元素,也能第一个处理消息,这与经典窗口的消息处理流程相悖。所以我通过让CCMenu与CCTableView通过建立窗口父子关系来达到解决第一个bug的目的。 1.首先派生出CCMenuNoMessage自CCMenu,并重载registerTouchDispatcher(),并上此函数为空,这样这个菜单实例是不会接受到任何消息的,因为我们的目的是建立父子窗口关系,所以应该完全有父窗口决定消息传递。并添加静态成员函数static CCMenuNoMessage * create(CCMenuItem* item, ...);在此函数的实现中,拷贝CCMenu内对应的代码,但需要改变里面的new()对象,CCMenuNoMessage *pRet = new CCMenuNoMessage();2.再派生出CCTableViewWindow自CCTableView,重载所有的ccTouch···消息处理函数,在ccTouchBegan中添加窗体点击处理, [cpp] view plaincopy01.//if not hit bounding box 02. if( false == boundingBox().containsPoint( pt ) ) 03. { 04. return false; 05. } 这样未点中TableView的消息是不处理。3.接着将消息传递给所有CCMenu子窗口。 [cpp] view plaincopy01.//hande message to menu items 02. m_bBeganForMenu = true; 03. for( int i = 0; i < m_pCellsUsed->count(); ++i ) 04. { 05. CCTableViewCell* pCell = ( CCTableViewCell* )m_pCellsUsed->objectAtIndex( i ); 06. if( pCell->getChildrenCount() > 0 ) 07. { 08. int iNumChildren = pCell->getChildren()->count(); 09. for( int iChildIndex = 0; iChildIndex < iNumChildren; ++iChildIndex ) 10. { 11. CCLayer* pItem = dynamic_cast< CCLayer* >( pCell->getChildren()->objectAtIndex( iChildIndex ) ); 12. 13. //if not a layer, it won't process touch messages. 14. if( !pItem ) 15. { 16. continue; 17. } 18. 19. //if this menu item is interested in this message, add it to array for other messages 20. if( true == pItem->ccTouchBegan( touch, event ) ) 21. { 22. m_pMenuItemsInterest->addObject( pItem ); 23. } 24. } 25. } 26. } 27. 28. CCScrollView::ccTouchBegan( touch, event); 通过标记变量来判断,此次用户点击是否要滑动(后面消息会用到),并将对点击消息感兴趣的CCMenu保留下来,让其接受ccTouchMove,ccTouchEnded消息。4.再处理ccTouchMoved消息, [cpp] view plaincopy01.//if this is the first ccTouchMove() message for menu itmes 02. if( m_bBeganForMenu ) 03. { 04. //prevent menu items processing other ccTouchMove() messages 05. m_bBeganForMenu = false; 06. for( int i = 0; i < m_pMenuItemsInterest->count(); ++i ) 07. { 08. CCLayer* pItem = ( CCLayer* )m_pMenuItemsInterest->objectAtIndex( i ); 09. //menu items doesn't process ccTouchMove(), cancel it. 10. 11. assert( NULL != pItem ); 12. pItem->ccTouchCancelled( touch, event ); 13. 14. } 15. 16. m_pMenuItemsInterest->removeAllObjects(); 17. } 18. 19. 20. CCScrollView::ccTouchMoved(touch, event); 如果ccTouchBegan之后紧接着是ccTouchMove消息,那么证明用户要滑动,此时,我们让对消息感兴趣的CCMenu释放掉。注意这里调用ccTouchCancelled消息,而非ccTouchEnded,因为后者会触发点击消息,这是我们不想看到的。5.再处理ccTouchEnded消息, [cpp] view plaincopy01.//if ccTouchEnded message is the next message after ccTouchBegan() message processed, 02. //it proves that user wants to select this menu item. 03. //then, let some menu item process this message. 04. if( m_bBeganForMenu ) 05. { 06. m_bBeganForMenu = false; 07. for( int i = 0; i < m_pMenuItemsInterest->count(); ++i ) 08. { 09. CCLayer* pItem = ( CCLayer* )m_pMenuItemsInterest->objectAtIndex( i ); 10. assert( NULL != pItem ); 11. pItem->ccTouchEnded( touch, event ); 12. } 13. 14. m_pMenuItemsInterest->removeAllObjects(); 15. } 16. 17. CCScrollView::ccTouchEnded(touch, event); 如果标记为真,证明用户的触摸未曾移动过就松开了,证明用户想点击次CCMenu,那么我们应该让对消息感兴趣的CCMenu完成点击消息。 通过这几步,我们建立起来了一个正常的父子关系,能让CCTableViewWindow与CCMenuNoMessage相处融洽,且解决了第一个bug。但第二个bug也可以称为非bug,或者系统级别的bug,这是由cocos2d的消息框架造成的,我们的CCTableViewWindow实在是势单力薄,无法在一个不融洽的体系内解决这个bug。所以CCTableViewWindow已经完成了它的使命,如果要解决第二个bug,我初步想法是自己建立一套层级消息管理器,让CCLayer能根据ZOrder顺序来处理消息。目前这个工作还没有让我做,我暂时只是个想法。 第二个bug的解决方案已经给出,详情将看cocos2d-x 建立自己的层级窗口消息机制。