laya2.0 box2d系列二 碰撞体

参考
官方文档 2D物理系统概述

一、基础概念

1.刚体rigidbody :刚体是指在运动中和受力作用后,形状和大小不变,而且内部各点的相对位置不变的物体。可参考laya2.0 box2d系列一 基础概念和刚体

2.碰撞体collider:碰撞体是给物体加一个判定框,当碰撞框重叠的时候,两物体发生碰撞。碰撞体是检测物理碰撞的框架,他永远跟随物体的刚体移动,不会产生偏差。碰撞体有四种:矩形碰撞体,圆形碰撞体,线形碰撞体,多边形碰撞体。每个碰撞体都是继承自碰撞体基类。

  • PolygonCollider->ColliderBase->Component 2D多边形碰撞体
  • CircleCollider->ColliderBase->Component 2D圆形碰撞体
  • ChainCollider->ColliderBase->Component 2D线形碰撞体
  • BoxCollider->ColliderBase->Component 2D矩形碰撞体
二、矩形碰撞体例子

1.施加力
laya2.0 box2d系列一 基础概念和刚体中介绍了刚体的许多特性,如果想用示例来演示,比如放一个静态的Sprite地面(设置一下texture,然后添加内部刚体类型static),然后一个sprite矩形盒子(设置一下texture,然后添加内部刚体类型dynamic,受重力影响)从空中落下停在地面上,如果不添加碰撞体,盒子会直接穿过地面,坠出屏幕(如果先添加碰撞体,IDE会自动添加刚体)。添加矩形刚体,IDE会把宽高设定为自动与父容器相同,然后就能看到盒子稳稳地停在地面上了。

private onKeyDown(e: Laya.Event): void {
    var rig: Laya.RigidBody = this.dropBox.getComponent(Laya.RigidBody);
    switch (e.keyCode) {
        case Laya.Keyboard.LEFT:
            // rig.setVelocity({ x: -1, y: 0 });
            //报错:this._body.applyForceToCenter is not a function
            //rig.applyForceToCenter({ x: -1, y: 0 });
            rig.applyForce(rig.body.GetWorldCenter(),{x:-1,y:0});

这里setVelocity是可以的,但是applyForceToCenter报错、applyForce无效,原因不明。applyLinearImpulseToCenter也是失效的。使用原生方式var box2d:any = window['box2d'];rig.body.ApplyForce(new box2d.b2Vec2(-1,0),rig.body.GetWorldCenter());也是不行。

2.改变重心,做出类似不倒翁的效果。这里laya没有封装SetMassData方法,需要自己写

var box2d: any = window['box2d'];
var rigid: Laya.RigidBody = this.dropBox.getComponent(Laya.RigidBody);
var md: any = new box2d.b2MassData();
rigid.body.GetMassData(md);
//重新设置重心
md.center.Set(1, 1.25);
rigid.body.SetMassData(md);

设置重心时,如果过低时,SetMassData会报错:0<a.I&&!this.m_flag_fixedRotationFlag&&(this.m_I=a.I-this.m_mass*box2d.b2Dot_V2_V2(a.center,a.center)这里如果a.I小于0,就会出问题。可以打开e_centerOfMassBit配置来观察重心的位置。

image.png

switch (e.keyCode) {
    case Laya.Keyboard.LEFT:
    rig.applyLinearImpulse({ x: 50, y: 50 }, { x: -1, y: 0 });

按下左箭头,就能看到不倒翁效果了。

三、laya封装

1.ColliderBase.as

/**@private 获取碰撞体信息*/
protected function getDef():* {
    if (!_def) {
        var def:* = new window.box2d.b2FixtureDef();
        def.density = density;
        def.friction = friction;
        def.isSensor = isSensor;
        def.restitution = restitution;
        def.shape = _shape;
        _def = def;
    }
    return _def;
}

/**
 * @private
 * 碰撞体参数发生变化后,刷新物理世界碰撞信息
 */
public function refresh():void {
    if (enabled && rigidBody) {
        var body:* = rigidBody.body;
        if (fixture) {
            //trace(fixture);
            if (fixture.GetBody() == rigidBody.body) {
                rigidBody.body.DestroyFixture(fixture);
            }
            fixture.Destroy();
            fixture = null;
        }
        var def:* = getDef();
        def.filter.groupIndex = rigidBody.group;
        def.filter.categoryBits = rigidBody.category;
        def.filter.maskBits = rigidBody.mask;
        fixture = body.CreateFixture(def);
        fixture.collider = this;
    }
}

override protected function _onEnable():void {
    rigidBody || Laya.systemTimer.callLater(this, _checkRigidBody);
}

private function _checkRigidBody():void {
    if (!rigidBody) {
        var comp:RigidBody = owner.getComponent(RigidBody);
        if (comp) {
            this.rigidBody = comp;
            refresh();
        }
    }
}

可以看到碰撞体和刚体之间的关联,是通过owner.getComponent来获得引用的。def.shape = _shape;则用来处理形状的。

2.BoxColider.as
这里显示是要覆盖上面提到的_shape了。使用了_shape.SetAsBox(_width / 2 / Physics.PIXEL_RATIO, _height / 2 / Physics.PIXEL_RATIO, new window.box2d.b2Vec2((_width / 2 + _x) / Physics.PIXEL_RATIO, (_height / 2 + _y) / Physics.PIXEL_RATIO));矩形只是多边形类的特殊情况,注意SetASBox方法传入的是半宽和半高,需要原始的width或height/2

override protected function getDef():* {
    if (!_shape) {
        _shape = _temp || (_temp = new window.box2d.b2PolygonShape());
        _setShape(false);
    }
    this.label = (this.label || "BoxCollider");
    return super.getDef();
}

IDE上的fitsize按钮,点一下这个按钮,碰撞体的大小就会自适应为节点宽高。

这里测试一下在HTML的CANVAS中,SetASBox和坐标的关系:

var fixDef = new b2FixtureDef;
fixDef.density = 1.0;//密度
fixDef.friction = 0.5;//摩擦
fixDef.restitution = 0.2;//弹性
fixDef.shape = new b2PolygonShape;//矩形
fixDef.shape.SetAsBox(2, 2);//宽高           
//创建一个矩形地板刚体
var bodyDef = new b2BodyDef;                 
bodyDef.type = b2Body.b2_staticBody;//静态的
bodyDef.position.x = 2;    //X轴
bodyDef.position.y = 2;    //Y轴     
//世界中添加一个刚体
world.CreateBody(bodyDef).CreateFixture(fixDef);

现在能看到一个矩形刚体,宽高应该都是4,然后位于坐标系的左上角,这时候position的x,y都是2。可以改改position的x,y就知道这个坐标对应的是矩形的中心坐标,并不是左上角。LAYA在封装时,就把这个偏移累加上去了:_shape.SetAsBox(_width / 2 / Physics.PIXEL_RATIO, _height / 2 / Physics.PIXEL_RATIO, new window.box2d.b2Vec2((_width / 2 + _x) / Physics.PIXEL_RATIO, (_height / 2 + _y) / Physics.PIXEL_RATIO));

3.CircleColider.as

override protected function getDef():* {
    if (!_shape) {
        _shape = _temp || (_temp = new window.box2d.b2CircleShape());
        _setShape(false);
    }
    this.label = (this.label || "CircleCollider");
    return super.getDef();
}
private function _setShape(re:Boolean = true):void {
    _shape.m_radius = _radius / Physics.PIXEL_RATIO;
    _shape.m_p.Set((_radius + _x) / Physics.PIXEL_RATIO, (_radius + _y) / Physics.PIXEL_RATIO);
    if (re) refresh();
}

同上面的矩形,也可以看看在HTML的CANVAS中圆心和坐标系的关系:

var fixDef2 = new b2FixtureDef;
fixDef2.density = 1.0;
fixDef2.friction = 0.5;
fixDef2.restitution = 0.2;
fixDef2.shape = new b2CircleShape(1);   

var bodyDef2 = new b2BodyDef;                
bodyDef2.type = b2Body.b2_dynamicBody;
bodyDef2.position.x = 1;    
bodyDef2.position.y = 0;    
world.CreateBody(bodyDef2).CreateFixture(fixDef2);

圆的半径是1,如果将圆心的x设置为0,就只能看到一半的圆了。只有将x设置为1,才恰好看到整个圆。这也说明,和矩形一样,postion的坐标对应的是圆心坐标。LAYA和矩形一样,也进行了处理:_shape.m_p.Set((_radius + _x) / Physics.PIXEL_RATIO, (_radius + _y) / Physics.PIXEL_RATIO);,也就是与编辑器中的_x,_y保持了一致,以左上角为0,0点。

4.ChainColider.as

/**用逗号隔开的点的集合,格式:x,y,x,y ...*/
private var _points:String = "0,0,100,0";
/**是否是闭环,注意不要有自相交的链接形状,它可能不能正常工作*/
private var _loop:Boolean = false;

override protected function getDef():* {
    if (!_shape) {
        _shape = _temp || (_temp = new window.box2d.b2ChainShape());
        _setShape(false);
    }
    this.label = (this.label || "ChainCollider");
    return super.getDef();
}

private function _setShape(re:Boolean = true):void {
    var arr:Array = _points.split(",");
    var len:int = arr.length;
    if (len % 2 == 1) throw "ChainCollider points lenth must a multiplier of 2";
    
    var ps:Array = [];
    for (var i:int = 0, n:int = len; i < n; i += 2) {
        ps.push(new window.box2d.b2Vec2((_x + parseInt(arr[i])) / Physics.PIXEL_RATIO,
         (_y + parseInt(arr[i + 1])) / Physics.PIXEL_RATIO));
    }
    _loop ? _shape.CreateLoop(ps, len / 2) : _shape.CreateChain(ps, len / 2);
    
    if (re) refresh();
}

5.PolygonCollider.as
2D多边形碰撞体,暂时不支持凹多边形,如果是凹多边形,先手动拆分为多个凸多边形。节点个数最多是b2_maxPolygonVertices,这数值默认是8,所以点的数量不建议超过8个,也不能小于3个。

/**用逗号隔开的点的集合,格式:x,y,x,y ...*/
private var _points:String = "50,0,100,100,0,100";

override protected function getDef():* {
    if (!_shape) {
        _shape = _temp || (_temp = new window.box2d.b2PolygonShape());
        _setShape(false);
    }
    this.label = (this.label || "PolygonCollider");
    return super.getDef();
}

private function _setShape(re:Boolean = true):void {
    var arr:Array = _points.split(",");
    var len:int = arr.length;
    if (len < 6) throw "PolygonCollider points must be greater than 3";
    if (len % 2 == 1) throw "PolygonCollider points lenth must a multiplier of 2";
    
    var ps:Array = [];
    for (var i:int = 0, n:int = len; i < n; i += 2) {
        ps.push(new window.box2d.b2Vec2((_x + parseInt(arr[i])) / Physics.PIXEL_RATIO,
            (_y + parseInt(arr[i + 1])) / Physics.PIXEL_RATIO));
    }
    
    _shape.Set(ps, len / 2);
    if (re) refresh();
}

注意在IDE中设置时,是顺时针方向旋转的。在编辑器中,在线上单击左键增加一个点,点可以拖拽,双击点会删除这个点。这里最后一个点会自动连接到起始点,形成一个封闭的多边形。

这里使用的Set方法是有两个限制的:

(1)凹多边形(相邻两个边之间夹角有一个大于180度)在进行碰撞模拟时会出现不可预知的错误。有一些工具类如b2Separator类可以处理,在拉小登博客 运行时创建多边形刚体有提到。

(2)在向vertices数组中添加顶点坐标时,要保证顶点的顺序是顺时针的。否则也会在碰撞中出现未知错误。

关于坐标,这里在HTML的CANVAS中再做个测试:

var fixDef3 = new b2FixtureDef;
var shape = new b2PolygonShape();
var arr = [];
arr.push(new b2Vec2(5,0));
arr.push(new b2Vec2(10,5));
arr.push(new b2Vec2(0,5));
shape.SetAsVector(arr,3);
fixDef3.shape = shape;
var bodyDef3 = new b2BodyDef;
bodyDef3.type = b2Body.b2_staticBody;//静态的
bodyDef3.position.x = 0;//X轴
bodyDef3.position.y = 10;//Y轴
world.CreateBody(bodyDef3).CreateFixture(fixDef3);

这里position.x=0,可以看到三角形在最左侧。也就是position对应的是多边形的最左侧,这与LAYA中的坐标系是一致的,所以不需要再对中心点,只加上偏移的_x,_y即可:ps.push(new window.box2d.b2Vec2((_x + parseInt(arr[i])) / Physics.PIXEL_RATIO, (_y + parseInt(arr[i + 1])) / Physics.PIXEL_RATIO));

6.SetAsOrientedBox
拉小登博客 信手绘制线条刚体中,是可以创建未封闭的多边形:

for each(var segment:Segment in segmentsArray) {
    var shapeBoxRequest:b2PolygonShape = new b2PolygonShape();
    shapeBoxRequest.SetAsOrientedBox(segment.length /2/ 30, 2 / 30, 
    new b2Vec2(segment.centerX/30, segment.centerY/30), segment.rotation);
    fixtureRequest.shape = shapeBoxRequest;
    body.CreateFixture(fixtureRequest);
}

使用的SetAsOrientedBox方法,可以在拉小登博客 Box2D多边形刚体中看到使用方式,参考API可知,四个参数分别是宽、高、偏移向量、旋转角度。但是在laya封装的physics.js中没有找到SetAsOrientedBox方法

四、属性

1.密度density:number
注意不要为了实现刚体不受重力影响,将密度设为0。系统会强制把0变成1.0。
2.摩擦力friction
和linearDamping有些相似,friction决定了两个刚体之间接触时所产生的摩擦力大小,而linearDamping影响的是刚体自由运动时速度的衰减,类似空气阻力。friction可以实现冰面、沙滩、玻璃等表面摩擦力不同的材质。
3.弹性restitution
反弹力,可以模拟橡胶、木板、钢铁等不同材质
4.碰撞筛选group,category,mask
box2d默认所有刚体进行碰撞模拟,filter就是将刚体分门别类,比如游戏中敌我之间进行碰撞,而敌人与敌人之间不进行碰撞检测。在源码中,会有一个shouldCollide函数来返回一个Boolean来确定是否碰撞:

var f1:b2FilterData = fixtureA.GetFilterData();
var f2:b2FilterData = fixtureB.GetFilterData

if(f1.groupIndex == f2.groupIndex && f1.groupIndex != 0){
    return f1.groupIndex > 0;
}

var collide:boolean = (f1.maskBits & f2.categoryBits) != 0 
&& (f1.categoryBits & f2.maskBits) != 0;

参考源码,当groupIndex相等时,如果大于0,则会碰撞;小于0,则永远不会碰撞;剩余情况就看mask和category了:在laya中,group默认为0,category默认为1,即0x0001;mask默认为-1,转二进制取反加1,也就是0xFFFF;

观察(f1.maskBits & f2.categoryBits) != 0 && (f1.categoryBits & f2.maskBits) != 0;简称前面的条件为a,后面的条件为b.因为&&的意义,想要碰撞,就要求a和b都不为0。

设置好filter属性后,如果想清除筛选条件,需要使用新建一个b2FilterData作为属性,直接置null会报错。laya是在refresh中刷新属性的:

var def:* = getDef();
def.filter.groupIndex = rigidBody.group;
def.filter.categoryBits = rigidBody.category;
def.filter.maskBits = rigidBody.mask;
fixture = body.CreateFixture(def);

5.isSensor
设置为true时,当与其他刚体发生碰撞后,只进行碰撞检测并派发碰撞事件,不进行碰撞模拟,通常与碰撞事件搭配使用,比如割绳子糖果落到气泡里上浮效果;默认为false,就是普通刚体,正常进行碰撞模拟。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容