我们使用TiledMap制作了一张简单的地图,并把它加入到了程序中,紧接着本章将实现地图的双指缩放和单指移动功能。

双指缩放,单指拖动的实现

在Cocos2d-x 3.x 中,实现触摸响应的一般流程如下:

  1. 重载触摸回调函数
  2. 创建并绑定触摸事件
  3. 实现触摸回调函数

具体实现如下:

1、首先,在GameScene.h文件中声明成员函数。

1
2
virtual void ontouchesBegan( const std::vector<cocos2d::Touch*>& touches,cocos2d::Event *event);
void ontouchesMoved( ottom:auto!important; float:none!important; height:auto!important; left:auto!important; line-height:1.1em!important; margin:0px!important; outline:0px!important; overflow:visible!important; padding:0px!important; position:static!important; right:auto!important; top:auto!important; vertical-align:baseline!important; width:auto!important; font-size:1em!important; min-height:inherit!important; color:rgb(0,cocos2d::Event *event);

2、在GameScene.cpp文件的init函数中创建并绑定触摸事件。

2
3
4
5
6
7
// 1 创建一个事件监听器
auto listener = EventListenerTouchAllAtOnce::create();
// 2 绑定触摸事件
listener->ontouchesBegan = CC_CALLBACK_2(GameScene::ontouchesBegan, this ); // 触摸开始时触发
listener->ontouchesMoved = CC_CALLBACK_2(GameScene::ontouchesMoved,0)!important; background:none!important">// 触摸移动时触发
// 3 添加监听器
_eventdispatcher->addEventListenerWithSceneGraPHPriority(listener,bgSprite);
  1. 在使用触摸事件时,我们首先需要创建一个事件监听器,事件监听器包含了触摸事件、键盘响应事件、加速记录事件、鼠标响应事件和自定义事件。其中的触摸监听类型触摸事件又分为 EventListenerTouchOneByOne(单点触摸) 和 EventListenerTouchAllAtOnce(多点触摸) 两种。
  2. 让监听器绑定事件处理函数。上面绑定的ontouchesBegan和ontouchesMoved分别响应的是触摸点击开始事件和移动事件。与之相关的还有onTouchEnded和onTouchCancelled两个事件处理函数,但目前我们的游戏还不需要(也有可能不会用到),所以这里就不用实现了。
  3. 监听器创建完成后需要把它绑定给_eventdispatcher事件分发器,_eventdispatcher 是 Node 的属性,通过它我们可以统一管理当前节点(如:场景、层、精灵等)的所有事件分发情况。
    将事件监听器 listener 添加到事件调度器_eventdispatcher中有两种方法,即如下的两个函数:
    1
    2
    void addEventListenerWithSceneGraPHPriority(EventListener* listener,Node* node);
    void addEventListenerWithFixedPriority(EventListener* listener, int fixedPriority);

    两者的主要区别在于它们加入到事件分发器中的优先级的差异。其中的使用 addEventListenerWithSceneGraPHPriority 方法添加的事件监听器优先级固定为0;而使用 addEventListenerWithFixedPriority 方法添加的事件监听器的优先级则可以自己设置,但不可以设置为 0,因为这个是保留给 SceneGraPHPriority使用的。

3、最后在GameScene.cpp文件中实现触摸回调函数
一旦玩家开始触碰屏幕,我们的程序就会开始调用相应的触摸事件处理函数来处理相应的逻辑,所以现在我们就可以来完成这部分的逻辑了。

实现中有以下几个需要注意的问题:

  • 需要判断触碰是单点还是多点,如果是多点,那么就缩放;是单点,就拖动。
  • 节点缩放的参考点默认是其锚点。显然,对于一个大地图背景来说,如果不实时改变它的锚点位置和本身位置,那它的缩放必然不会按照选取的区域进行缩放,必然会出现类似下图的情况。

(上图中,背景图片的锚点在蓝点的位置,当我们想放大红圈所圈的那棵树时,如果只是简单的改变背景的放大倍率,那一定会出现上图的第二种情况(目标会向右上角偏移);但如果我们把背景的锚点和位置都设置到目标处,那就会像第三中情况一样,得到一个比较好的放大效果。)

  • 当缩放到一定程度,如缩小到与可视区域一样时,为了避免出现空白的区域,我们需要做一些处理。同时地图不能无止境的放大或缩小,需要有一定的范围来约束。比如,放大到它本身的4倍时,应该停止放大。
  • 拖动地图移动时,地图不能移出可视区域,这里需要做边界控制。

掌握了这些注意事项以后,现在我们就可以开始具体的行动了。

首先,在GameScene.h中定义如下的变量:

2
Sprite* bgSprite;
Vec2 bgOrigin;

bgSprite是地图背景,需要缩放和移动的对象都是其子节点,这样我们就可以通过操作它来实现缩放和移动了。bgOrigin用于记录bgSprite的初始原点位置。

接着,我们跳转到GameScene.cpp的init()方法,修改之前添加地图背景的方法,同时初始化bgOrigin。如下代码所示:

7
8
9
10
11
12
13
14
15
16
17
18
19
mapLayer = Layer::create();
this ->addChild(mapLayer,-1);
bgSprite = Sprite::create( "2.jpg" );
bgSprite->setAnchorPoint(Vec2::ZERO);
bgSprite->setPosition(Vec2::ZERO),
bgOrigin = Vec2(Vec2::ZERO);
mapLayer->addChild(bgSprite);
auto treeSprite = Sprite::create( "1.png" );
treeSprite->setAnchorPoint(Vec2::ZERO);
treeSprite->setPosition(Vec2::ZERO),
treeSprite->setScale(2);
bgSprite->addChild(treeSprite,2);
auto map = TMXTiledMap::create( "mymap4.tmx" );
map->setAnchorPoint(Vec2::ZERO);
map->setPosition(Vec2::ZERO),
bgSprite->addChild(map,1);

因为对层而言,它相比于其他的节点来说,其锚点、位置、大小都不好控制,所以我们需要通过另外的节点(比如这里的bgSprite)来执行后面的缩放和移动等动作。

最后压轴来了,实现触摸事件的处理函数如下:

19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
void GameScene::ontouchesMoved( const std::vector<Touch*>& touches,Event *event)
{
auto winSize = Director::getInstance()->getWinSize();
if (touches.size() > 1) // 多点进行缩放
{
// 得到当前两触摸点
auto point1 = touches[0]->getLocation();
auto point2 = touches[1]->getLocation();
// 计算两点之间得距离
auto currdistance = point1.distance(point2);
// 计算两触摸点上一时刻之间得距离
auto prevdistance = touches[0]->getPrevIoUsLocation().distance(touches[1]->getPrevIoUsLocation());
// 两触摸点与原点的差向量,pointVec1和pointVec2是相对于bgSprite的位置
auto pointVec1 = point1 - bgOrigin;
auto pointVec2 = point2 - bgOrigin;
// 两触摸点的相对中点
auto relMidx = (pointVec1.x + pointVec2.x) / 2 ;
auto relMidy = (pointVec1.y + pointVec2.y) / 2 ;
// 计算bgSprite的锚点
auto anchorX = relMidx / bgSprite->getBoundingBox().size.width;
auto anchorY = relMidy / bgSprite->getBoundingBox().size.height;
// 相对屏幕的中点
auto absMidx = (point2.x + point1.x) / 2 ;
auto absMidy = (point2.y + point1.y) / 2 ;
// 缩放时,为了避免出现空白的区域,需要做以下的边界处理。
// 当bgSprite快要进入到屏幕时,修改bgSprite的位置(既absMidx和absMidy)。
if ( bgOrigin.x > 0)
{
absMidx -= bgOrigin.x;
}
if ( bgOrigin.x < -bgSprite->getBoundingBox().size.width + winSize.width )
{
absMidx += -bgSprite->getBoundingBox().size.width + winSize.width - bgOrigin.x;
}
if ( bgOrigin.y > 0 )
{
absMidy -= bgOrigin.y;
}
if ( bgOrigin.y < -bgSprite->getBoundingBox().size.height + winSize.height )
{
absMidy += -bgSprite->getBoundingBox().size.height + winSize.height - bgOrigin.y;
}
// 重设bgSprite锚点和位置
bgSprite->setAnchorPoint(Vec2(anchorX,anchorY));
bgSprite->setPosition(Vec2(absMidx,absMidy));
// 根据两触摸点前后的距离计算缩放倍率
auto scale = bgSprite->getScale() * ( currdistance / prevdistance);
// 控制缩放倍率在1~4倍之间,最小倍率不能太小,不让背景将不能填充满整个屏幕。
scale = MIN(4,MAX(1,scale));
bgSprite->setScale(scale);
// 更新原点位置
bgOrigin = Vec2(absMidx,absMidy) - Vec2(bgSprite->getBoundingBox().size.width * anchorX,bgSprite->getBoundingBox().size.height * anchorY) ;
}
else if (touches.size() == 1) // 单点进行移动
{
// 单点时,touches中只有一个Touch对象,所以通过touches[0]就可以得到触摸对象
auto touch = touches[0];
// 计算滑动过程中的滑动增量
auto diff = touch->getDelta();
// 得到当前bgSprite的位置
auto currentPos = bgSprite->getPosition();
// 得到滑动后bgSprite应该所在的位置
auto pos = currentPos + diff;
// 得到此刻bgSprite的尺寸
auto bgSpriteCurrSize = bgSprite->getBoundingBox().size;
//边界控制,约束pos的位置
pos.x = MIN(pos.x,bgSpriteCurrSize.width * bgSprite->getAnchorPoint().x);
pos.x = MAX(pos.x,-bgSpriteCurrSize.width + winSize.width + bgSpriteCurrSize.width * bgSprite->getAnchorPoint().x);
pos.y = MIN(pos.y,bgSpriteCurrSize.height * bgSprite->getAnchorPoint().y);
pos.y = MAX(pos.y,-bgSpriteCurrSize.height + winSize.height + bgSpriteCurrSize.height * bgSprite->getAnchorPoint().y);
// 重设bgSprite位置
bgSprite->setPosition(pos);
// 更新原点位置
if ( pos.x >= bgSpriteCurrSize.width * bgSprite->getAnchorPoint().x
|| pos.x <= -bgSpriteCurrSize.width + winSize.width + bgSpriteCurrSize.width * bgSprite->getAnchorPoint().x)
{
diff.x = 0;
}
if ( pos.y >= bgSpriteCurrSize.height * bgSprite->getAnchorPoint().y
|| pos.y <= -bgSpriteCurrSize.height + winSize.height + bgSpriteCurrSize.height * bgSprite->getAnchorPoint().y)
{
diff.y = 0;
}
bgOrigin += diff;
}
}

以上就是ontouchesMoved函数的实现方法了,原理已在注释中解释清楚,所以我想理解起来已经不会很难。下面给出一张示意图帮助大家理解:

下图是缩放过程中刚好出现空白的区域时的图形示意图:

此时空白的区域的宽等于-bgSprite->getBoundingBox().size.width + winSize.width - bgOrigin.x,所以我们把背景的位置向右移动-bgSprite->getBoundingBox().size.width + winSize.width - bgOrigin.x个单位就可以避免这种情况的出现。

代码中有一点需要注意的是,在缩放过程中,bgSprite的尺寸不断变化的,所以计算起锚点或进行边界处理时,一定要用它缩放后的尺寸宽高来计算,而不能是它本身的宽高。 所以代码中计算bgSprite的尺寸我们用getBoundingBox函数来获得经过缩放和旋转之后的外框盒大小,而不用getContentSize函数来获得节点原始的大小。

iOS端多点触碰默认是关闭的,所以需要在AppController.mm 程序启动回调中启用多点触摸才可以,具体方法是在以下的函数段后加入[eaglView setMultipletouchEnabled:YES];

如下所示:

9
CCEAGLView *eaglView = [CCEAGLView viewWithFrame: [window bounds]
pixelFormat: kEAGLColorFormatRGBA8
depthFormat: GL_DEPTH24_STENCIL8_OES
preserveBackbuffer: NO
sharegroup: nil
multiSampling: NO
numberOfSamples: 0];
[eaglView setMultipletouchEnabled:YES];

总的来说,要想很好的实现这一功能不是容易的,以上就是我们实现了的一种方法,虽然细节上还有一些问题,也未在真机上测试,但还是希望能对大家的学习有所帮助。如果你有更好的方法实现,也可以提出来,大家一起进步学习。

基于Cocos2d-x的实现tilemap地图双指缩放,单指拖动的更多相关文章

  1. ios – 链接点击监听器上的WKWebView?

    在WKWebView类中是否存在类似onLinkClickListener的东西?我试着谷歌搜索但没有发现任何东西,我也发现了一些关于simillar类型的stackoverflow的未解答的问题.我需要一个linkClickListener的原因是,当我点击链接并且页面尚未加载时,它不会加载网站.当页面加载了监听器时,我也可以创建一个花哨的加载屏幕.解决方法你可以这样做将WKNavigation

  2. 多个监听器用于委托iOS

    我有一个带有代理didSelectString的类搜索栏.我有一个实现委托的A类和一个实现委托的B类.但是只有来自A类的代理才被执行.代表可以有多个监听器吗?并且如何实现这一点解决方法该委托是单一消息传递协议.如果要发送更改的多个对象,则需要使用NSNotifications.您可以使用通知中心传递对象,如下所示:想要收听通知时并设置选择器

  3. 早期Swift中Cocos2D初始化代码的重构

    但是遗憾的是Swift2.2中还是不支持Type的class属性关键字,只能用static,我们期待Swift3的改进吧!

  4. swift中提供属性监听器,可以监听属性的改变

  5. [swift]-属性监听器

    1:swift中监听属性的变化是通过属性监听器来监听OC中监听属性变化是通过set方法来监听属性变化2:代码演示意见反馈邮件:1415429879@qq.com欢迎你们的阅读和赞赏、谢谢!

  6. Swift - 属性监听器

    属性监听器介绍:属性监听器,监听属性的值改变,就像按钮的点击事件一样来监听其他的值改变举例:监听scrollView的contentOffset属性的改变

  7. 手把手教你swift项目添加cocos2dx-lua

    去倒杯水吧,这个过程会很久。。。至此,工程已经全部配置完毕,你已经解决一个大Boss了;今天就到这里吧,之后我们再续如何用swift调用lua手把手教你swift项目添加cocos2dx-lua

  8. 手把手教你swift项目集成cocos2dx-js模块

    前几天在swift项目中集成了Lua模块,使得在swift工程中用Lua写游戏逻辑成为了可能,具体工程及配置见手把手教你swift项目添加cocos2dx-lua,由于公司最近要把js做的小游戏集成到iOS原生应用中,于是我们将解锁另外一个场景,好了,下面开始;同样的,首先你得有一个swift项目我们从头开始,建立一个swift项目;我们默认你已经可以自己创建一个全新的swift项目了,这很简单,不是么?

  9. android – 使用回调/监听器链接RxJava observable

    这是我到目前为止:解决方法调整clemp6r的解决方案,这是另一个既不需要主题也不需要嵌套订阅的解决方案:一般来说,我认为总是可以使用Observable.create()在Observable中包装任何基于回调的异步操作.

  10. Android – 可靠地获取当前位置

    错了,因为如果用户的位置已经稳定,那么我的听众永远不会被调用,因为位置没有改变.但GPS将一直运行,直到我的听众被移除,耗尽电池……获取当前位置的正确方法是什么,而不会误认为当前位置的旧位置?我不介意等几分钟.编辑:有可能我错误的是没有被叫的听众,它可能只需要比我想象的要长一点……

随机推荐

  1. 【cocos2d-x 3.x 学习笔记】对象内存管理

    Cocos2d-x的内存管理cocos2d-x中使用的是上面的引用计数来管理内存,但是又增加了一些自己的特色。cocos2d-x中通过Ref类来实现引用计数,所有需要实现内存自动回收的类都应该继承自Ref类。下面是Ref类的定义:在cocos2d-x中创建对象通常有两种方式:这两中方式的差异可以参见我另一篇博文“对象创建方式讨论”。在cocos2d-x中提倡使用第二种方式,为了避免误用第一种方式,一般将构造函数设为protected或private。参考资料:[1]cocos2d-x高级开发教程2.3节[

  2. 利用cocos2dx 3.2开发消灭星星六如何在cocos2dx中显示中文

    由于编码的不同,在cocos2dx中的Label控件中如果放入中文字,往往会出现乱码。为了方便使用,我把这个从文档中获取中文字的方法放在一个头文件里面Chinese.h这里的tex_vec是cocos2dx提供的一个保存文档内容的一个容器。这里给出ChineseWords,xml的格式再看看ChineseWord的实现Chinese.cpp就这样,以后在需要用到中文字的地方,就先include这个头文件然后调用ChineseWord函数,获取一串中文字符串。

  3. 利用cocos2dx 3.2开发消灭星星七关于星星的算法

    在前面,我们已经在GameLayer中利用随机数初始化了一个StarMatrix,如果还不知道怎么创建星星矩阵请回去看看而且我们也讲了整个游戏的触摸事件的派发了。

  4. cocos2dx3.x 新手打包APK注意事项!

    这个在编译的时候就可以发现了比较好弄这只是我遇到的,其他的以后遇到再补充吧。。。以前被这两个问题坑了好久

  5. 利用cocos2dx 3.2开发消灭星星八游戏的结束判断与数据控制

    如果你看完之前的,那么你基本已经拥有一个消灭星星游戏的雏形。开始把剩下的两两互不相连的星星消去。那么如何判断是GameOver还是进入下一关呢。。其实游戏数据贯穿整个游戏,包括星星消除的时候要加到获得分数上,消去剩下两两不相连的星星的时候的加分政策等,因此如果前面没有做这一块的,最好回去搞一搞。

  6. 利用cocos2dx 3.2开发消灭星星九为游戏添加一些特效

    needClear是一个flag,当游戏判断不能再继续后,这个flag变为true,开始消除剩下的星星clearSumTime是一个累加器ONE_CLEAR_TIME就是每颗星星消除的时间2.连击加分信息一般消除一次星星都会有连击信息和加多少分的信息。其实这些combo标签就是一张图片,也是通过控制其属性或者runAction来实现。源码ComboEffect.hComboEffect.cpp4.消除星星粒子效果消除星星时,为了实现星星爆裂散落的效果,使用了cocos2d提供的粒子特效引擎对于粒子特效不了

  7. 02 Cocos2D-x引擎win7环境搭建及创建项目

    官网有搭建的文章,直接转载记录。环境搭建:本文介绍如何搭建Cocos2d-x3.2版本的开发环境。项目创建:一、通过命令创建项目前面搭建好环境后,怎样创建自己的Cocos2d-x项目呢?先来看看Cocos2d-x3.2的目录吧这就是Cocos2d-x3.2的目录。输入cocosnew项目名–p包名–lcpp–d路径回车就创建成功了例如:成功后,找到这个项目打开proj.win32目录下的Hello.slnF5成功了。

  8. 利用cocos2dx 3.2开发消灭星星十为游戏添加音效项目源码分享

    一个游戏,声音也是非常的重要,其实cocos2dx里面的简单音效引擎的使用是非常简单的。我这里只不过是用一个类对所有的音效进行管理罢了。Audio.hAudio.cpp好了,本系列教程到此结束,第一次写教程如有不对请见谅或指教,谢谢大家。最后附上整个项目的源代码点击打开链接

  9. 03 Helloworld

    程序都有一个入口点,在C++就是main函数了,打开main.cpp,代码如下:123456789101112131415161718#include"main.h"#include"AppDelegate.h"#include"cocos2d.h"USING_NS_CC;intAPIENTRY_tWinMain{UNREFERENCED_ParaMETER;UNREFERENCED_ParaMETER;//createtheapplicationinstanceAppDelegateapp;return

  10. MenuItemImage*图标菜单创建注意事项

    学习cocos2dx,看的是cocos2d-x3.x手游开发实例详解,这本书错误一大把,本着探索求知勇于发现错误改正错误的精神,我跟着书上的例子一起调试,当学习到场景切换这个小节的时候,出了个错误,卡了我好几个小时。

返回
顶部