Cocos2dx3.0过渡篇(三)触摸机制Cocos2dx3.0过渡篇(三)触摸机制

admin 2018-3-27 182

<iframe id="cproIframe_u1834336" width="468" height="15" src="http://pos.baidu.com/acom?adn=0&amp;at=231&amp;aurl=&amp;cad=1&amp;ccd=24&amp;cec=UTF-8&amp;cfv=16&amp;ch=0&amp;col=zh-CN&amp;conOP=0&amp;cpa=1&amp;dai=2&amp;dis=0&amp;ltr=http%3A%2F%2Fwww.baidu.com%2Fs%3Fwd%3Dcocos2dx3.0%2B%25E8%25A7%25A6%25E6%2591%25B8%26rsv_spt%3D1%26issp%3D1%26f%3D3%26rsv_bp%3D0%26rsv_idx%3D2%26ie%3Dutf-8%26tn%3Dbaiduhome_pg%26rsv_enter%3D1%26inputT%3D7872%26oq%3Dbase%2Bclass%2Bha&amp;ltu=http%3A%2F%2F88cto.com%2Fpanjing%2Farticle%2Fdetails%2F2370.html&amp;lunum=6&amp;n=57014038_cpr&amp;pcs=1370x782&amp;pis=10000x10000&amp;ps=308x427&amp;psr=1440x900&amp;pss=1370x1333&amp;qn=cb9ea12776cfd0f2&amp;rad=&amp;rsi0=468&amp;rsi1=15&amp;rsi5=4&amp;rss0=%23FFFFFF&amp;rss1=%23FFFFFF&amp;rss2=%230000ff&amp;rss3=&amp;rss4=&amp;rss5=&amp;rss6=%23e10900&amp;rss7=&amp;scale=&amp;skin=&amp;td_id=1834336&amp;tn=tlink_default_468_15&amp;tpr=1418981657789&amp;ts=1&amp;xuanting=0&amp;dtm=BAIDU_DUP2_SETJSONADSLOT&amp;dc=2&amp;di=u1834336" align="center,center" marginwidth="0" marginheight="0" scrolling="no" frameborder="0" allowtransparency="true" style="box-sizing: border-box; margin: 0px; padding: 0px; list-style: none; max-width: 100%;"></iframe>

尊重原创,转载请注明来自:star特530的CSDN博客 http://blog.csdn.net/start530/article/details/18325493

本来在中午休息时间打算大展拳脚,好好写一篇新触摸机制相关的博文,结果,等真正下手的时候才发现无从下手,很多地方自己都说不清,赶紧看了下testCpp,才发现原来是这样,还可以这样,哦?这样都行?哎,我还是太年轻了。

 

咱也只能挑简单的讲了。

假设要实现拖动一个精灵移动,那我们的步骤是:

1、 创建一个精灵sprite

2、一个触摸事件 listener ,设置listener的onTouchBegan,onTouchMoved,onTouchEnded;

3、将sprite 和 listener关联起来。

 

实现如下:

1、 创建精灵:

1Point origin = Director::getInstance()->getVisibleOrigin();
2Size size = Director::getInstance()->getVisibleSize();
3 
4auto sprite = Sprite::create("Images/CyanSquare.png");
5sprite->setPosition(origin+Point(size.width/2, size.height/2) + Point(-80, 80));
6addChild(sprite, 1);

2、 创建 listener

1auto listener1 = EventListenerTouchOneByOne::create();//创建一个触摸监听
2listener1->setSwallowTouches(true);//设置是否想下传递触摸

01//3.0 后可以直接在touchBegan后添加它的实现代码,而不用特意去写一个touchBegan的函数
02listener1->onTouchBegan = [](Touch* touch, Event* event){
03auto target = static_cast<Sprite*>(event->getCurrentTarget());//获取的当前触摸的目标
04     
05Point locationInNode = target->convertToNodeSpace(touch->getLocation());
06Size s = target->getContentSize();
07Rect rect = Rect(0, 0, s.width, s.height);
08     
09if (rect.containsPoint(locationInNode))//判断触摸点是否在目标的范围内
10      return true;
11else
12      return false;
13};
14 
15 //拖动精灵移动
16listener1->onTouchMoved = [](Touch* touch, Event* event){
17    auto target = static_cast<Sprite*>(event->getCurrentTarget());
18    target->setPosition(target->getPosition() + touch->getDelta());
19};
20 
21listener1->onTouchEnded = [=](Touch* touch, Event* event){
22};
23//将触摸监听添加到eventDispacher中去
24_eventDispatcher->addEventListenerWithSceneGraphPriority(listener1 ,sprite);
1}

以上就是移动一个精灵的实现过程,这里特意交代一些细节:

1)触摸监听listener的创建方式有两种,一种是:EventListenerTouchOneByOne,另一种是:EventListenerTouchAllAtOnce,顾名思义,EventListenerTouchOneByOne的意思单点触摸,EventListenerTouchAllAtOnce,是多点触摸,而不需要再用设置Delegate的方式来做了。3.0触摸机制还有个不同的地方,只要是放在最上面的那个精灵,那它的触摸优先级就最高。我们用的按钮Menu 就是用这种方式设置触摸优先级的。而

2)将listener1添加到事件调度中,这里用的是:

1_eventDispatcher->addEventListenerWithSceneGraphPriority(listener1 sprite);

我们进入addEventListenerWithSceneGraphPriority的定义中看一下,有下面这一行代码:

1listener->setFixedPriority(0);

它将精灵的触摸优先级设置成0,从这里我们可以引申出两个问题,一个就是当我们要给精灵设置触摸优先级时,

1listener->setFixedPriority(0);
,因为0已经被“官府”征用了,另一个问题就是:如果自己想设置精灵的触摸优先级,那应该怎么做呢?下面是提供的另外一种添加listener的方法:

1_eventDispatcher->addEventListenerWithFixedPriority(listener1 ,fixedPriority);

在第二个参数里设置触摸优先级,这样就可以了。

3)如果你有多个精灵sprite,且这些精灵都想实现拖动的功能,那么这些精灵都可以使用listener1这一个触摸监听,例如我们有三个精灵,sprite,sprite2,sprite3,他们调用listener1的方式:

1_eventDispatcher->addEventListenerWithSceneGraphPriority(listener1, sprite1);
2 _eventDispatcher->addEventListenerWithSceneGraphPriority(listener1->clone(), sprite2);
3_eventDispatcher->addEventListenerWithSceneGraphPriority(listener1->clone(), sprite3);

其中sprite2和sprite3都是克隆了listener1的,进入clone()的定义,我们看到以下代码:

01EventListenerTouchOneByOne* EventListenerTouchOneByOne::clone()
02{
03    auto ret = new EventListenerTouchOneByOne();
04    if (ret && ret->init())
05    {
06        ret->autorelease();
07         
08        ret->onTouchBegan = onTouchBegan;
09        ret->onTouchMoved = onTouchMoved;
10        ret->onTouchEnded = onTouchEnded;
11        ret->onTouchCancelled = onTouchCancelled;
12         
13        ret->_claimedTouches = _claimedTouches;
14        ret->_needSwallow = _needSwallow;
15    }
16    else
17    {
18        CC_SAFE_DELETE(ret);
19    }
20    return ret;
21}

以上代码主要的目的也就是实现克隆touchbegan,touchmoved,touchended。

3、删除触摸监听

如果想移除sprite的触摸移动,可以这么做:

1_eventDispatcher->removeEventListeners(EventListener::Type::TOUCH_ONE_BY_ONE);

这样就OK了。

 

好了,先说到这里吧。今晚公司尾牙请客,喝了蛮多酒的,所以这篇博文写的可能不够周密,望大家见谅。

3.0新的地方讲的也差不多了,简单的就不多说了,难的我也不懂。所以呢,就这样吧。接下来应该是写一些关于3.0的例子吧。恩。


有人问:3.0bate版本 的 继承layer的LayerColor,想停止LayerColor的触摸调用而使用setTouchEnabled,编译器提示声明被否决,肿么办?有神马替代函数能够停止触摸

 答:将setTouchEnable(),换成setEnable();试试


最新回复 (0)
返回