OBB碰撞检测,坐标点逆时针

class OBBTest extends egret.DisplayObjectContainer {
private obb1:OBB;
private obb2:OBB; private points1:Array<egret.Point>;
private points2:Array<egret.Point>;
private sprite1:egret.Sprite;
private sprite2:egret.Sprite;
private resPoint:egret.Point = new egret.Point();
constructor() {
super(); this.touchEnabled = true;
var stage = egret.MainContext.instance.stage;
stage.addEventListener(egret.TouchEvent.TOUCH_BEGIN, this.onTouchBegin, this);
stage.addEventListener(egret.TouchEvent.TOUCH_MOVE, this.onTouchMove, this);
stage.addEventListener(egret.TouchEvent.TOUCH_END, this.onTouchEnd, this); this.points1 = [new egret.Point(0,0),new egret.Point(100,200),new egret.Point(300,300),new egret.Point(300,100),new egret.Point(0,0)];
this.sprite1 = this.drawSprite(this.points1);
this.addChild(this.sprite1);
this.sprite1.x = 100;
this.sprite1.y = 100; this.points2 = [new egret.Point(0,0),new egret.Point(100,200),new egret.Point(300,300),new egret.Point(300,100),new egret.Point(0,0)];
this.sprite2 = this.drawSprite(this.points2);
this.addChild(this.sprite2);
this.sprite2.x = 100;
this.sprite2.y = 400; //创建两obb盒子
this.obb1 = new OBB();
this.obb2 = new OBB(); egret.startTick(this.onEnterFrame, this);
} private pt:egret.Point = new egret.Point(0,0);
private onEnterFrame():boolean { for (var i = 0; i < this.points1.length; i++) {
var point1:egret.Point = this.points1[i];
this.sprite1.localToGlobal(point1.x, point1.y, this.pt);
this.obb1.setVertex(i, this.pt.x, this.pt.y);
} for (var j = 0; j < this.points2.length; j++) {
var point2:egret.Point = this.points2[j];
this.sprite2.localToGlobal(point2.x, point2.y, this.pt);
this.obb2.setVertex(j, this.pt.x, this.pt.y);
} this.sprite1.visible = !this.obb1.isCollidWithOBB(this.obb2); return false;
} private drawSprite(points:Array<egret.Point>):egret.Sprite {
var startPoint:egret.Point = points[0];
var sprite = new egret.Sprite();
sprite.graphics.clear();
sprite.graphics.beginFill( 0xffffff, 0.5 );
sprite.graphics.moveTo( startPoint.x, startPoint.y );
for (var i = 1; i < points.length ; i++) {
var point = points[i];
sprite.graphics.lineTo( point.x, point.y );
}
sprite.graphics.endFill(); return sprite;
} private onTouchBegin(evt:egret.TouchEvent):void {
var posX = evt.stageX;
var posY = evt.stageY; this.sprite2.x = posX;
this.sprite2.y = posY;
} private onTouchMove(evt:egret.TouchEvent):void {
var posX = evt.stageX;
var posY = evt.stageY; this.sprite2.x = posX;
this.sprite2.y = posY;
} private onTouchEnd(evt:egret.TouchEvent):void {
var posX = evt.stageX;
var posY = evt.stageY; this.sprite2.x = posX;
this.sprite2.y = posY;
}
}
/**
* 投影判断
*/
class Projection {
private min:number = 0;
private max:number = 0;
constructor(min, max) {
this.min = min;
this.max = max;
} public getMin(){
return this.min;
} public getMax(){
return this.max;
} /**
* 判断是否叠加
*/
public overlap(proj:Projection):boolean {
if(this.min > proj.getMax()) return false;
if(this.max < proj.getMin()) return false; return true;
}
};
class OBB {
private pList:Array<any> = [] //保存盒子顶点数
constructor() {
this.pList = [];
} public getVertex(idx):egret.Point {
if (idx >= this.pList.length) {
return null;
} return this.pList[idx];
}; public setVertex(idx:number, x:number, y:number):void {
if (!this.pList[idx]) {
this.pList[idx] = new egret.Point(0, 0);
} this.pList[idx].x = x;
this.pList[idx].y = y;
} /**
* 计算分离轴
* 如果边向量为(x,y),那么法向量为(-y,x),你也可以设置为(y,-x)。结果没有什么变化。
*/
public getAxies():Array<egret.Point> {
var axies = [];
var len = this.pList.length;
var pOut = new egret.Point(0,0);
for(var i = 0; i < len; i++) {
var startPoint:egret.Point = this.pList[i];
var endPoint:egret.Point = this.pList[(i + 1) % len];
var point = startPoint.subtract(endPoint); var length = Math.sqrt(point.x * point.x + point.y * point.y);
pOut.x = point.x / length;
pOut.y = point.y / length; axies[i] = new egret.Point();
axies[i].x = -pOut.y ;
axies[i].y = pOut.x ;
}
return axies;
} /**
* 计算投影
* 计算出一个投影线条
* 只是保存了两个float形的数据,分别表示OBB盒在分离轴上投影的最小值和最大值
*/
public getProjection(axies:egret.Point):Projection {
var min = this.pList[0].x * axies.x + this.pList[0].y * axies.y;
var max = min ; var len = this.pList.length;
for (var i = 1; i < len; i++) {
var temp = this.pList[i].x * axies.x + this.pList[i].y * axies.y;
if (temp > max) {
max = temp;
} else if (temp < min) {
min = temp;
}
}// end for return new Projection(min, max);
} //对传递进来的OBB判断是否与调用这个方法的OBB发生了交叉
public isCollidWithOBB(obb:OBB):boolean {
//获取分离轴
var axies1 = this.getAxies();
var axies2 = obb.getAxies(); var p1:Projection = null;
var p2:Projection = null; //Check for overlap for all of the axies
for(var i = 0; i < axies1.length; i++)
{
p1 = this.getProjection(axies1[i]);
p2 = obb.getProjection(axies1[i]);
if(!p1.overlap(p2))
{
return false ;
}
} for(var j = 0; j < axies2.length; j ++)
{
p1 = this.getProjection(axies2[j]);
p2 = obb.getProjection(axies2[j]);
if(!p1.overlap(p2))
{
return false ;
}
} return true ;
}
}

测试,在Main中添加

var obb = new OBBTest();
this.addChild(obb);

OBB碰撞的更多相关文章

  1. Cocos2d-x教程(34)-三维物体OBB碰撞检測算法

    欢迎增加Cocos2d-x 交流群:193411763 个中心点.1个旋转矩阵和3个1/2边长(注:一个旋转矩阵包括了三个旋转轴,若是二维的OBB包围盒则是一个中心点,两个旋转轴,两个1/2边长). ...

  2. 游戏中的2D OBB碰撞模型的碰撞算法介绍和实践

    前言 上一篇博文说道,射线与场景中模型上的所有三角形求交时,会大幅度影响效率且花费比较多的时间,因此会采取使用包围盒的形式,进行一个加速求交.在此文中介绍OBB碰撞模型的碰撞算法 OBB的碰撞模型 有 ...

  3. 2D空间的OBB碰撞实现

    OBB全称Oriented bounding box,方向包围盒算法.其表现效果和Unity的BoxCollider并无二致.由于3D空间的OBB需要多考虑一些情况 这里仅关注2D空间下的OBB. 实 ...

  4. 游戏碰撞OBB算法(java代码)

    业务需求      游戏2D型号有圆形和矩形,推断说白了就是碰撞检测 :      1.圆形跟圆形是否有相交      2.圆形跟矩形是否相交       3.矩形和矩形是否相交           ...

  5. AABB包围盒、OBB包围盒、包围球的比較

    1) AABB 包围盒: AABB 包围盒是与坐标轴对齐的包围盒, 简单性好, 紧密性较差(尤其对斜对角方向放置的瘦长形对象, 採用AABB, 将留下非常大的边角空隙, 导致大量不是必需的包围盒相交測 ...

  6. AABB边框、OBB边框、通过比较球包围

    1) AABB 包围盒: AABB 包围盒是与坐标轴对齐的包围盒, 简单性好, 紧密性较差(尤其对斜对角方向放置的瘦长形对象, 採用AABB, 将留下非常大的边角空隙, 导致大量不是必需的包围盒相交測 ...

  7. 矩形旋转碰撞,OBB方向包围盒算法实现

    怎样进行2D旋转矩形的碰撞检測.能够使用一种叫OBB的检測算法(Oriented bounding box)方向包围盒.这个算法是基于SAT(Separating Axis Theorem)分离轴定律 ...

  8. [算法][包围盒]球,AABB,OBB

    参考地址请看图片水印:http://www.cnblogs.com/iamzhanglei/archive/2012/06/07/2539751.html http://blog.sina.com.c ...

  9. 3D有向包围盒与球体碰撞的算法

    之前发的小游戏滚蛋躲方块中,用它来判断球体与立方体是否发生了碰撞. http://www.cnblogs.com/WhyEngine/p/3350012.html 现在发布下该算法: 有向包围盒OBB ...

随机推荐

  1. Codeforces 812E Sagheer and Apple Tree ——(阶梯博弈)

    之前在bc上做过一道类似的阶梯博弈的题目,那题是移动到根,这题是移动到叶子.换汤不换药,只要和终态不同奇偶的那些位置做nim即可.因此这题给出了一个条件:所有叶子深度的奇偶性相同.同时需要注意的是,上 ...

  2. POI的XWPFTableCell的方法

    1. XWPFParagraph addParagraph() 在这个表格单元格中添加一个段落 2. void addParagraph(XWPFParagraph p) 给这个表格加一段 3. ja ...

  3. 算法-java实现

    1. 质因数分解 public static List<Integer> factorize(int n){ List<Integer> factors = new Array ...

  4. maven的pom报错web.xml is missing and <failOnMissingWebXml> is set to true

    错误信息:web.xml is missing and <failOnMissingWebXml> is set to true 解决办法:https://blog.csdn.net/si ...

  5. Pluck CMS 4.7.10远程代码执行漏洞分析

    本文首发于先知: https://xz.aliyun.com/t/6486 0x01漏洞描述 Pluck是用php编写的一款的小型CMS影响版本:Pluck CMS Pluck CMS 4.7.10( ...

  6. C# Under the Hood: async/await (Marko Papic)

    https://www.markopapic.com/csharp-under-the-hood-async-await/ Async and await keywords came with C# ...

  7. Flutter移动电商实战 --(28)列表页_商品列表后台接口调试

    主要调试商品列表页的接口 这个接口是最难的因为有大类.小类还有上拉加载 先配置接口 config/service_url.dart //const serviceUrl='http://test.ba ...

  8. 安装RabbitMQ管理插件失败

    运行 rabbitmq-plugins.bat enable rabbitmq_management后提示失败信息  是因为erlang和RabbitMQ版本冲突导致

  9. Python3中_和__的用途和区别

    访问可见性问题 对于上面的代码,有C++.Java.C#等编程经验的程序员可能会问,我们给Student对象绑定的name和age属性到底具有怎样的访问权限(也称为可见性).因为在很多面向对象编程语言 ...

  10. Linux 远程登陆图形界面

    如果我的解决方案帮助到了你,请随手点亮一颗小红心.如有疑问,可在下方评论区处留言. 利用Xmanager,linux启用XDMCP协议(可直接修改配置文件,也可以采用在Xshell中运行gdmconf ...