cocos2d-x for wp 之Box2D游戏-是男人就坚持60M(一)
搞了几天。发现cocos2d-x对c#(xna)真的是抛弃很彻底了。以后还是转c++吧。
废话结束,开始记录。
首先,跟上一节一样,先创建好一个世界并让这个世界开始模拟物理世界。
创建一个继承于CCLayer的一个层:PlayGame。并添加好下面三个引用、一个常量以及一些全局变量:
1 using cocos2d;2 using Box2D.XNA;3 using Microsoft.Xna.Framework;4 5 6 public static double PTM_RATIO = 32.0;7 8 World world;9 CCSprite ball; 10 Body groundBody;
重写init函数、node函数
![](/assets/blank.gif)
![](/assets/blank.gif)
1 public override bool init() 2 { 3 if (!base.init()) 4 return false; 5 //获取窗口大小 6 CCSize winSize = CCDirector.sharedDirector().getWinSize(); 7 8 9 10 CCLabelTTF title = CCLabelTTF.labelWithString("Boxing", "Arial", 24); 11 title.position = new CCPoint(winSize.width / 2, winSize.height / 2 - 50); 12 this.addChild(title, 1); 13 14 Vector2 gravity = new Vector2(0.0f, 0.0f); 15 bool doSleep = true; 16 world = new World(gravity, doSleep); 17 18 19 BodyDef groundBodyDef = new BodyDef(); 20 groundBodyDef.position = new Vector2(0, 0); 21 groundBody = world.CreateBody(groundBodyDef); 22 PolygonShape groundBox = new PolygonShape();//凸多边形 23 FixtureDef boxShapeDef = new FixtureDef(); 24 boxShapeDef.shape = groundBox; 25 groundBox.SetAsEdge(new Vector2(0, 0), new Vector2((float)(winSize.width / PTM_RATIO), 0)); 26 groundBody.CreateFixture(boxShapeDef); 27 groundBox.SetAsEdge(new Vector2(0, 0), new Vector2(0, (float)(winSize.height / PTM_RATIO))); 28 groundBody.CreateFixture(boxShapeDef); 29 groundBox.SetAsEdge(new Vector2(0, (float)(winSize.height / PTM_RATIO)), 30 new Vector2((float)(winSize.width / PTM_RATIO), (float)(winSize.height / PTM_RATIO))); 31 groundBody.CreateFixture(boxShapeDef); 32 groundBox.SetAsEdge(new Vector2((float)(winSize.width / PTM_RATIO), (float)(winSize.height / PTM_RATIO)), 33 new Vector2((float)(winSize.width / PTM_RATIO), 0)); 34 groundBody.CreateFixture(boxShapeDef); 35 36 Body body1 = createBall(); 37 Body body2 = createBall(); 38 39 Player play1 = new Player(this,world,groundBody); 40 41 this.schedule(tick); 42 return true; 43 }
![](/assets/blank.gif)
![](/assets/blank.gif)
1 public static new CCLayer node() 2 { 3 PlayGame layer = new PlayGame(); 4 if (layer.init()) 5 { 6 return layer; 7 } 8 else 9 layer = null; 10 return layer; 11 }
基本上和上一节的内容差不多,但是我们这里把重力gravity设置为0,0.因为我们这个游戏并不需要重力的存在。 至于createBall是一个自定义函数,而Player是一个自定义的类,等下会讲到。
createBall函数:
![](/assets/blank.gif)
![](/assets/blank.gif)
1 public Body createBall() 2 { 3 Body body; 4 Random rand = new Random(); 5 ball = CCSprite.spriteWithFile("images/ball"); 6 this.addChild(ball); 7 int Vx = rand.Next(1, 7) * 100; 8 int Vy = rand.Next(1, 4) * 100; 9 BodyDef ballBodyDef = new BodyDef(); 10 ballBodyDef.type = BodyType.Dynamic;//动态类型 11 ballBodyDef.position = new Vector2((float)(Vx / PTM_RATIO), (float)(Vy / PTM_RATIO)); 12 ballBodyDef.userData = ball; 13 14 body = world.CreateBody(ballBodyDef); 15 CircleShape circle = new CircleShape(); 16 circle._radius = (float)(26.0 / PTM_RATIO); 17 18 FixtureDef ballShapeDef = new FixtureDef(); 19 ballShapeDef.shape = circle; 20 ballShapeDef.density = 1.0f;//密度 21 ballShapeDef.friction = 0.0f;//摩擦 22 ballShapeDef.restitution = 1.0f; 23 body.CreateFixture(ballShapeDef); 24 25 int Fx = rand.Next(1, 7) * 10; 26 int Fy = rand.Next(1, 4) * 10; 27 Vector2 force = new Vector2(Fx, Fy); 28 body.ApplyLinearImpulse(force, ballBodyDef.position); 29 30 return body; 31 }
这个函数的内容也是老内容了,除了设置了一个冲力force。
1 Vector2 force = new Vector2(Fx, Fy);
2 body.ApplyLinearImpulse(force, ballBodyDef.position);
有了这个冲力,我们的游戏中初始化的球就会自己动起来
添加好一个tick函数
![](/assets/blank.gif)
![](/assets/blank.gif)
1 void tick(float dt) 2 { 3 world.Step(dt, 10, 10); 4 5 for (Body b = world.GetBodyList(); b != null; b = b.GetNext()) 6 { 7 if (b.GetUserData() != null) 8 { 9 CCSprite ballData = (CCSprite)b.GetUserData(); 10 ballData.position = new CCPoint((float)(b.GetPosition().X * PTM_RATIO), 11 (float)(b.GetPosition().Y * PTM_RATIO)); 12 ballData.rotation = -1 * MathHelper.ToDegrees(b.GetAngle()); 13 } 14 } 15 }
以上都是一些老内容。不熟悉可以去看上一篇介绍。
最后修改一下程序入口applicationDidFinishLaunching
1 CCScene pScene = CCScene.node(); 2 pScene.addChild(Classes.PlayGame.node()); 3 pDirector.deviceOrientation = ccDeviceOrientation.CCDeviceOrientationLandscapeLeft; 4 pDirector.runWithScene(pScene); 5 return true;
运行,程序就会出现两个小球不停的在屏幕里来回跳动。
接下来,是该实现控制小球和各个小球之间的碰撞了。
构造Player自定义类。继承于CCSprite、ICCTargetedTouchDelegate。因为我们要手动(触屏)控制这个玩家,所以也要同时继承于ICCTargetedTouchDelegate。
这里也是我fengyun1989不同的地方。在他那边是直接在游戏中的游戏界面写touch事件(因为Layer里也有touch事件)。而我习惯于将各个对象分开。
首先,添加一些变量。方便后面调用,具体用处在用到时细说。
MouseJoint mouseJoint=null; Fixture ballFixture; Body spriteBody; World locWord; Body gBody; public static double PTM_RATIO = 32.0;
重写构造函数Player()
![](/assets/blank.gif)
![](/assets/blank.gif)
1 public Player(CCLayer layer,World world,Body groundBody)2 {3 gBody = groundBody;4 locWord = world;5 6 CCSize s = CCDirector.sharedDirector().getWinSize();7 this.initWithFile("images/ball");8 this.position = new CCPoint(s.width / 2, s.height / 2);9 layer.addChild(this); 10 addBoxBodyForSprite(this, world); 11 }
同时将PlayGame里创建好的世界world和groundBody传过来,因为我们的玩家必须在同一个世界里才行。而且后面构造一些fixture也需要用到。
接下来,是该进行触摸事件的修改了。先是ccTouchBegan()
![](/assets/blank.gif)
![](/assets/blank.gif)
1 public virtual bool ccTouchBegan(CCTouch touch, CCEvent eventer) 2 { 3 //创建一个鼠标关节 4 if (mouseJoint != null) 5 return false; 6 CCPoint location = touch.locationInView(touch.view()); 7 location = CCDirector.sharedDirector().convertToGL(location); 8 Vector2 locationWorld = new Vector2((float)(location.x / PTM_RATIO), (float)(location.y / PTM_RATIO)); 9 10 if (ballFixture.TestPoint(locationWorld)) 11 { 12 MouseJointDef md = new MouseJointDef(); 13 md.bodyA = gBody; 14 md.bodyB = spriteBody; 15 md.collideConnected = true; 16 md.target = locationWorld; 17 md.maxForce = 1000.0f * spriteBody.GetMass(); 18 mouseJoint = (MouseJoint)locWord.CreateJoint(md); 19 spriteBody.SetAwake(true); 20 return true; 21 } 22 else 23 return false;
首先,我们把touch坐标转换成coocs2d坐标(convertToGL)然后,再转换成Box2d坐标(locationWorld)。
如果是的话,我们就创建一个所谓的”鼠标关节“。在Box2d里面,一个鼠标关节用来让一个body朝着一个指定的点移动---在这里个例子中,就是用户点的方向。
当你创建一个mousejoint后,你赋值给它两个body。第一个没有被使用,通常都是设置成groundbody(前面提到这里用到了)。第二个,就是你想让它移动的body(这个类里面的精灵的body),在这个例子中就是spriteBody。
你指定移动的终点---这个例子中就是用户点击的位置locationWorld。
然后,设置bodyA和bodyB碰撞的时候,把它当成是碰撞,而不是忽略它。这个很重要!
如果没有设置,当我们用鼠标拖动这个你的小球的时候,它并不会与屏幕的边界相碰撞。
指定移动body的最大的力是多少。如果你减少这个数值的话,spritebody响应鼠标移动时就会慢一些。但是,我们想让spritebody快速地响应鼠标的变化。
最后,我们把这个关节加入到world中。同时,我们还要把body设置成苏醒的(awake)。之所以要这么做,是因为如果body在睡觉的话,那么它就不会响应鼠标的移动!
上面用到了一个鼠标关节mouseJoint。这是一个自定义的类。内容如下:
![](/assets/blank.gif)
![](/assets/blank.gif)
1 class MyContact 2 { 3 public Fixture fixtureA; 4 public Fixture fixtureB; 5 6 } 7 class MyContactListener : IContactListener 8 { 9 public List<MyContact> contacts = new List<MyContact>(); 10 11 public void BeginContact(Contact contact) 12 { 13 MyContact myContact = new MyContact() 14 { 15 fixtureA = contact.GetFixtureA(), 16 fixtureB = contact.GetFixtureB() 17 }; 18 contacts.Add(myContact); 19 20 } 21 22 public void EndContact(Contact contact) 23 { 24 contacts.Clear(); 25 } 26 27 public void PostSolve(Contact contact, ref ContactImpulse impulse) 28 { 29 } 30 public void PreSolve(Contact contact, ref Manifold oldManifold) 31 { 32 } 33 }
接下来,让我们添加ccTouchesMoved方法:
![](/assets/blank.gif)
![](/assets/blank.gif)
1 public virtual void ccTouchMoved(CCTouch touch, CCEvent eventer) 2 { 3 if (mouseJoint == null) 4 return; 5 Vector2 point = new Vector2(this.convertTouchToNodeSpace(touch).x, 6 this.convertTouchToNodeSpace(touch).y); 7 CCPoint location = touch.locationInView(touch.view()); 8 location = CCDirector.sharedDirector().convertToGL(location); 9 Vector2 locationWorld = new Vector2((float)(location.x / PTM_RATIO), (float)(location.y / PTM_RATIO)); 10 11 mouseJoint.SetTarget(locationWorld); 12 }
我们更新了鼠标关节的目标位置(也就是我们想让玩家移动的位置的)。
我们添加ccTouchesCacelled和ccTouchesEnded方法:这里只实现一件事,就是在我们移动完paddle或者取消移动之后销毁mouse joint。
![](/assets/blank.gif)
![](/assets/blank.gif)
1 public virtual void ccTouchEnded(CCTouch touch, CCEvent eventer) 2 { 3 if (mouseJoint != null) 4 { 5 mouseJoint = null; 6 locWord.DestroyJoint(mouseJoint); 7 } 8 } 9 10 public void ccTouchCancelled(CCTouch touch, CCEvent eventer) 11 { 12 if (mouseJoint != null) 13 { 14 mouseJoint = null; 15 locWord.DestroyJoint(mouseJoint); 16 17 } 18 }
编译并运行,你现在可以用手指控制属于你的精灵了,同时可以让它与篮球相互碰撞!
转载于:https://www.cnblogs.com/dieaz5/archive/2013/03/24/2979436.html
cocos2d-x for wp 之Box2D游戏-是男人就坚持60M(一)相关推荐
- (译)如何使用cocos2d来制作简单的iphone游戏:更猛的怪物和更多的关卡。(第三部分。完!)...
免责申明(必读!):本博客提供的所有教程的翻译原稿均来自于互联网,仅供学习交流之用,切勿进行商业传播.同时,转载时不要移除本申明.如产生任何纠纷,均与本博客所有人.发表该翻译稿之人无任何关系.谢谢合作 ...
- (译)如何使用cocos2d来制作一个打地鼠的游戏:第一部分
免责申明(必读!):本博客提供的所有教程的翻译原稿均来自于互联网,仅供学习交流之用,切勿进行商业传播.同时,转载时不要移除本申明.如产生任何纠纷,均与本博客所有人.发表该翻译稿之人无任何关系.谢谢合作 ...
- Java敌人跟地图的碰撞_(译)加入敌人和战斗:如果使用cocos2d制作基于tiled地图的游戏:第三部分...
免责申明(必读!):本博客提供的所有教程的翻译原稿均来自于互联网,仅供学习交流之用,切勿进行商业传播.同时,转载时不要移除本申明.如产生任何纠纷,均与本博客所有人.发表该翻译稿之人无任何关系.谢谢合作 ...
- (译)如何使用cocos2d制作基于tile地图的游戏教程:第一部分
免责申明(必读!):本博客提供的所有教程的翻译原稿均来自于互联网,仅供学习交流之用,切勿进行商业传播.同时,转载时不要移除本申明.如产生任何纠纷,均与本博客所有人.发表该翻译稿之人无任何关系.谢谢合作 ...
- 译)如何使用cocos2d制作基于tile地图的游戏教程:第一部分
(译)如何使用cocos2d制作基于tile地图的游戏教程:第一部分 免责申明(必读!):本博客提供的所有教程的翻译原稿均来自于互联网,仅供学习交流之用,切勿进行商业传播.同时,转载时不要移除本申明. ...
- 14. 如何使用cocos2d来制作一个打地鼠的游戏:第一部分
免责申明(必读!):本博客提供的所有教程的翻译原稿均来自于互联网,仅供学习交流之用,切勿进行商业传播.同时,转载时不要移除本申明.如产生任何纠纷,均与本博客所有人.发表该翻译稿之人无任何关系.谢谢合作 ...
- 飞机躲避小游戏---是男人就撑100秒的制作
飞机躲避小游戏---是男人就撑100秒的制作 EmilMatthew(EmilMatthew@126.com) 摘要: 可以将这个游戏的整体运作看成一个粒子系统,再加上子弹和飞机的碰撞判定即可.简单 ...
- Java小游戏——是男人就坚持20秒
Java小游戏--是男人就坚持20秒 给大家分享一个小游戏,这是一个Swing项目,先用代码画界面板,然后依次添加飞机.子弹.计时器等: 代码如下: /** 程序的主窗体*/ public class ...
- cocos2d-x for wp 之Box2D的应用
本文大部分内容是基于fengyun1989的博文,自己只是加深巩固而已 Box2d是一款用于2d游戏的物理引擎.在这个世界里创造出的物体都更接近于真实世界的物体. 首先,在项目添加Box2D.XNA. ...
最新文章
- iOS 动态更换icon
- Linux下动态链接库调用
- k8s 使用helm部署dashboard
- 用aspect在springboot中记录操作日志至数据库的详细过程
- SAP Spartacus B2B页面Budget页面的设计原理
- .NET 6 Preview 3 中 ASP.NET Core 的更新和改进
- mac 安装nodejs_阿里开源——用于前端和nodejs的轻量级任务管理和构建工具Dawn
- 分布式认知在计算机应用系统,人机交互作业
- 金数据一个不错的调查平台
- 孙宇晨终于和巴菲特共进晚餐 还赠送数字币作为见面礼
- Oracle数据库一致性读的原理
- MultCloud – 支持数据互传的网盘管理
- php抓取dom处理后数据,PHP简单DOMDocument抓取排除td类
- 计算机信息学中比较大小的代码,信息学奥赛计算机基础知识.doc
- 独立同分布(iid)随机变量的一些趣题
- cookie和seseion的区别和联系点
- 写给小白看的马尔科夫链(Markov Chain)最佳入门教程
- 爬取搜狗微信文章笔记2
- 苹果技术支持联系方式
- OpenCV 透射变换