CCTableView 与 CCMenu巧妙结合

简介: 今天被赋予了一项新的任务,那就是看看cocos2d新版本的CCTableView能不能解决项目TableView的Bug。 项目内的TableView会存在如下bug: 当TableView里面的MenuItem滚动出View方框时,用户应该不能在View方框外选中该MenuItem的。

今天被赋予了一项新的任务,那就是看看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中添加窗体点击处理,
    //if not hit bounding box
        if( false == boundingBox().containsPoint( pt ) )
        {
            return false;
        }


    这样未点中TableView的消息是不处理。
  3. 接着将消息传递给所有CCMenu子窗口。

    //hande message to menu items
        m_bBeganForMenu = true;
        for( int i = 0; i < m_pCellsUsed->count(); ++i )
        {
            CCTableViewCell* pCell = ( CCTableViewCell* )m_pCellsUsed->objectAtIndex( i );
            if( pCell->getChildrenCount() > 0 )
            {
                int iNumChildren = pCell->getChildren()->count();
                for( int iChildIndex = 0; iChildIndex < iNumChildren; ++iChildIndex )
                {
                    CCLayer* pItem = dynamic_cast< CCLayer* >( pCell->getChildren()->objectAtIndex( iChildIndex ) );
                    
                    //if not a layer, it won't process touch messages.
                    if( !pItem )
                    {
                        continue;
                    }
                    
                    //if this menu item is interested in this message, add it to array for other messages
                    if( true == pItem->ccTouchBegan( touch, event ) )
                    {
                        m_pMenuItemsInterest->addObject( pItem );
                    }
                }
            }
        }
        
        CCScrollView::ccTouchBegan( touch, event);


    通过标记变量来判断,此次用户点击是否要滑动(后面消息会用到),并将对点击消息感兴趣的CCMenu保留下来,让其接受ccTouchMove,ccTouchEnded消息。
  4. 再处理ccTouchMoved消息,

    //if this is the first ccTouchMove() message for menu itmes
        if( m_bBeganForMenu )
        {
            //prevent menu items processing other ccTouchMove() messages
            m_bBeganForMenu = false;
            for( int i = 0; i < m_pMenuItemsInterest->count(); ++i )
            {
                CCLayer* pItem = ( CCLayer* )m_pMenuItemsInterest->objectAtIndex( i );
                //menu items doesn't process ccTouchMove(), cancel it.
    
                assert( NULL != pItem );
                pItem->ccTouchCancelled( touch, event );
    
            }
            
            m_pMenuItemsInterest->removeAllObjects();
        }
        
        
        CCScrollView::ccTouchMoved(touch, event);


    如果ccTouchBegan之后紧接着是ccTouchMove消息,那么证明用户要滑动,此时,我们让对消息感兴趣的CCMenu释放掉。注意这里调用ccTouchCancelled消息,而非ccTouchEnded,因为后者会触发点击消息,这是我们不想看到的。
  5. 再处理ccTouchEnded消息,

    //if ccTouchEnded message is the next message after ccTouchBegan() message processed,
        //it proves that user wants to select this menu item.
        //then, let some menu item process this message.
        if( m_bBeganForMenu )
        {
            m_bBeganForMenu = false;
            for( int i = 0; i < m_pMenuItemsInterest->count(); ++i )
            {
                CCLayer* pItem = ( CCLayer* )m_pMenuItemsInterest->objectAtIndex( i );
                assert( NULL != pItem );
                pItem->ccTouchEnded( touch, event );
            }
            
            m_pMenuItemsInterest->removeAllObjects();
        }
        
        CCScrollView::ccTouchEnded(touch, event);



    如果标记为真,证明用户的触摸未曾移动过就松开了,证明用户想点击次CCMenu,那么我们应该让对消息感兴趣的CCMenu完成点击消息。
通过这几步,我们建立起来了一个正常的父子关系,能让CCTableViewWindow与CCMenuNoMessage相处融洽,且解决了第一个bug。但第二个bug也可以称为非bug,或者系统级别的bug,这是由cocos2d的消息框架造成的,我们的CCTableViewWindow实在是势单力薄,无法在一个不融洽的体系内解决这个bug。所以CCTableViewWindow已经完成了它的使命,如果要解决第二个bug,我初步想法是自己建立一套层级消息管理器,让CCLayer能根据ZOrder顺序来处理消息。目前这个工作还没有让我做,我暂时只是个想法。

第二个bug的解决方案已经给出,详情将看 cocos2d-x 建立自己的层级窗口消息机制。
相关文章
|
7月前
|
算法
数据结构与算法1.2 算法的定义 什么是好的算法 复杂度的渐进表示
数据结构与算法1.2 算法的定义 什么是好的算法 复杂度的渐进表示
37 0
|
3月前
|
Go
IsNil() 和 IsValid() 的精妙运用
IsNil() 和 IsValid() 的精妙运用
36 0
|
8月前
|
C语言
C数组超细节精髓
C数组超细节精髓
50 0
|
6月前
|
算法 程序员 C#
谈谈算法的基本思想
谈谈算法的基本思想
34 0
|
8月前
|
移动开发 网络虚拟化
【五讲四美】之“讲思想”
发挥一点工匠精神,对一个技术组内小运营需求的精进优化过程。
54 0
【五讲四美】之“讲思想”
|
9月前
|
存储 缓存 开发者
深入探讨递归方法:理解原理与优化技巧
递归是一种常见的编程技巧,它在解决问题时能够简化代码结构,提高可读性。然而,递归也容易导致性能问题和内存溢出等隐患。本文将深入探讨递归方法的原理,讨论递归的优化技巧,以帮助开发者更好地理解和应用递归。
170 1
|
10月前
|
算法 C语言
【算法】一篇文章弄清楚KMP算法的实现
【算法】一篇文章弄清楚KMP算法的实现
63 0
|
10月前
|
存储 编译器 C语言
【C】函数真的难嘛?其实一点也不难,原理很简单。
# 什么是函数 程序是由多个零件组合而成的,而函数就是这种“零件”的一个较小单位。 ## main函数和库函数 C语言程序中,main函数是必不可少的。程序运行的时候,会执行main函数的主题部分。main函数中使用了printf、scanf、puts等函数。由C语言提供的这些为数众多的函数称为库函数。 ## 什么是函数 当然,我们也可以自己创建函数。而实际上,我们也必须亲自动手创建各种函数。下面我们来自己创建一个简单的函数。 创建一个函数,接收两个整数参数,返回较大整数的值。 printf函数和scanf函数等创建得比较好得函数,即使不知道其内容,只要了解使用方法,也可以轻松使用。 ## 函
|
11月前
|
编解码 算法
关于位运算的巧妙性:小乖,你真的明白吗?
关于位运算的巧妙性:小乖,你真的明白吗?
|
算法
算法排序问题。每种排序代表着没中思考问题的方式。我们学习了选择排序,冒泡排序,归并排序。让我们去回顾回顾吧。重在思想的领悟。
算法排序问题。每种排序代表着没中思考问题的方式。我们学习了选择排序,冒泡排序,归并排序。让我们去回顾回顾吧。重在思想的领悟。
65 0
算法排序问题。每种排序代表着没中思考问题的方式。我们学习了选择排序,冒泡排序,归并排序。让我们去回顾回顾吧。重在思想的领悟。

热门文章

最新文章