Cocos2d-x 基础示例-精灵移动2

Cocos2d-x 基础示例-精灵移动2

 

先推荐一下红哥的精灵教程

http://blog.csdn.net/honghaier/article/details/8117903

 

红哥里面使用的ccTouchesMoved,回忆下我之前的教程,使用的是ccTouchesBegan,大家可以体验一下这两种方式的不同。

当我们的手指触碰到屏幕的一瞬间,调用的是ccTouchesBegan,之后手指如果在屏幕上滑动,手指会一直调用ccTouchesMoved,所以大家可以想象一下这两种方式不同。我的方式会无视之后手指的位置,精灵只会向第一个点击的地方移动。而红哥的方式则是无视第一点触碰的坐标,一直会跟着之后的手指位置移动,这就是 began(起始)和moved(滑动)的区别。

     但实际上,我们在正常实现触碰+移动的时候,很少使用ccMove家族的成员,最重要的原因是他们非常死板,很不灵活,除了贝塞尔(CCBezier)同学比较受欢迎,其他的同学基本不能和ccTouch家族结合使用。所以接下来,我们会模拟两种比较实用的方式来实现飞机的移动。

1. 精灵以匀速向当前触摸点行动,当触摸取消时,精灵停止。

这种方式相对而言比较常见,让我们一起实现之。

第一步:让我们在ccTouchesMoved函数中添加一下代码:

    CCSetIteratorit=pTouches->begin();

    CCTouch* touch=(CCTouch*)(*it);   

    CCPointm_tBeginPos=touch->locationInView();

   m_tBeginPos=CCDirector::sharedDirector()->convertToGL(m_tBeginPos);

 

等!!!

这里插播个话题,这段代码反复出现(ccTouch家族都用他),功能完全一样!如果大家发现这种倒霉代码,最好考虑把他变成一个更可爱的子函数。

我们在helloworld头文件中加入函数声明:

    // 触摸相关

CCPoint convertToGL(CCSet *pTouches);

并且一定要保证这位同学跟ccTouches那三兄弟在一起,这才是良好的编程习惯!!不然等代码量上去了,有你哭的那天。

   之后的实现如下

CCPoint HelloWorld::convertToGL(cocos2d::CCSet *pTouches)

{

    if (!pTouches) {

        CCSetIteratorit=pTouches->begin();

        CCTouch* touch=(CCTouch*)(*it);

        CCPointm_tBeginPos=touch->locationInView();

        m_tBeginPos=CCDirector::sharedDirector()->convertToGL(m_tBeginPos);

        return m_tBeginPos;

       

    }

    assert(!"pTouches is Null!!!!!");

}

部分同学对这里面的if (!pTouches)和assert(!"pTouches is Null!!!!!"),这是本人编写子函数的一个习惯,这样如果传进来的pTouches参数为空,编译时会直接捕捉到这个错误,而不会继续下传。当然以我们现在这个规模的程序来说,确实是有点杀鸡牛刀了,但我还是推荐大家以后编写子函数的时候使用这种方式

   这样这段代码就封装好了,以后我们编写哪个ccTouch的时候,可以直接使用我们convertToGL了!(这个函数的中文意思是“转换成GL”)

 

于是我们的ccTouchesMoved函数变成了这样

void HelloWorld::ccTouchesMoved(CCSet *pTouches, CCEvent *pEvent)

{

    CCPoint m_CurrentPos=convertToGL(pTouches);

 

}

我们现在获取了当前坐标,该让player向目标移动了!

第一:先定义飞机向那儿移动的速度。

在头文件中增加

private:

    float playerSpd=1;

定义playerSpd为速度,初始值为1。

第二:找出当前帧移动的单位向量

这里普及一下基础的几何知识,一个从点A和点B的向量AB,它是B点的坐标对应减掉A点的坐标。例如A点坐标是(a1,a2),B点坐标是(b1,b2),那么AB向量为(b1-a1,b2-a2).

但是我们单求“向量”是不对的,我们应该求的是“单位向量”。这个单位向量非常神奇,A点坐标加上(单位向量乘以两点间距离)之后,得出的结果是B点坐标。而速度则是每帧移动的距离,则(单位向量*速度)就可以计算出每帧的点坐标值啦!大家可以理解这个“单位向量”为“方向”。

为了实现这一步,我们可以定义一个子函数来实现上述过程,问题就在于这个子函数应该定义在什么地方。

其实可以思考一下,我们什么时候会用到这个子函数?所有存在CCPoint的情况下!

所以这个子函数最好定义在CCPoint类的类内。

另外一方面,为了能够让过程和数学过程更像,我们需要重载CCPoint的减号“-”,让这个函数支持两个CCPoint类做减法。

So,在CCPoint类的实现中中添加如下两个函数体(声明我就不写了,懒,大家自己加到头文件里吧)

CCPoint& CCPoint::operator- (const CCPoint& other)

{

    this->x= this->x-other.x;

    this->y= this->y-other.y;

    return *this;

}

另外增加计算单位向量的函数

CCPoint CCPoint::computeUnitVector()

{

    CCPoint p1=*this;

    assert(sqrt((p1.x)*(p1.x)+(p1.y)*(p1.y)));

    p1.x=p1.x/sqrt((p1.x)*(p1.x)+(p1.y)*(p1.y));

    p1.y=p1.y/sqrt((p1.x)*(p1.x)+(p1.y)*(p1.y));

    return p1;

}上面这个函数直接计算出某向量的单位向量了.

现在CCPoint类已经不单纯是一个“点”了,它现在还可以表示一个“向量”和“单位向量”,所以我们还需要用到让CCPoint类能够乘以一个speed。

在CCPoint头文件中加入定义,并在在CCPoint实现文件中,加入如下代码:

 

CCPoint& CCPoint::operator*(const float& speed)

{

    this->x=this->x*speed;

    this->y=this->y*speed;

    return *this;   

}

 

另一方面,还需要重载CCPoint的+号函数,能让两个CCPoint相加,得出一个新坐标,具体实现如下

CCPoint& CCPoint::operator+(const CCPoint &other)

{

    this->x= this->x+other.x;

    this->y= this->y+other.y;

    return *this;

}

OK,让我们回到CCTouchesMoved中;加入如下代码

void HelloWorld::ccTouchesMoved(CCSet *pTouches, CCEvent *pEvent)

{

    CCPoint m_CurrentPos=convertToGL(pTouches);

    //获得当前坐标

    CCPoint playerCurPos=player->getPosition();

    //如果距离不为0,则向那个方位移动一个速度

    if (ccpDistance(playerCurPos,m_CurrentPos)!=0)

    {

   

        CCPointm_vector=m_CurrentPos-playerCurPos;

        //m_vector转换为单位向量

        m_vector=m_vector.computeUnitVector();

        //每次调用ccTouchesMoved的时候,都向单位方向移动一个速度值

        player->setPosition(playerCurPos+m_vector*playerSpd);

 

    }

 

}

现在,当玩家在屏幕上滑动时,飞机就会向触摸的地方移动了。但有一个非常残酷的事实… 现在只有滑动的时候,才会调用CCTouchesMoved - -,CCTouchesMoved对对触摸进行了判断,所以现在这架飞机变成了非常奇异的家伙,你滑动手指,才会向你移动,你不滑动手指,他坚决不动.

很遗憾~苹果在底层做了判断,毕竟,如果手一直按住都响应的话,那设备将及其费电- -,所以我们只能换另外一种方式了~

可以仔细思考一下我们需求的流程,如下:

点击屏幕,则屏幕向该点以一定速度移动。

如果中途发生滑动,则修改该点坐标

如果触摸停止,则此动作停止

所以我们可以采用另外一种方式来实现此功能

ccTouchesBegan用来初始化点坐标,并激活飞机的一个动作,此动作不停得向触摸点移动

ccTouchesMoved用来改变此点的坐标

ccTouchesEnded终结此动作。

我们如果需要实现此功能,则需要以上三个函数协同工作,才能共同达成目标,下一讲将会讲述如何实现此功能。

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值