2020-06-12 08:47:13 +08:00
|
|
|
|
abstract class Collider extends Component{
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/** 对撞机的基本形状 */
|
2020-06-12 08:47:13 +08:00
|
|
|
|
public shape: Shape;
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/** 在处理冲突时,physicsLayer可以用作过滤器。Flags类有帮助位掩码的方法。 */
|
2020-06-12 08:47:13 +08:00
|
|
|
|
public physicsLayer = 1 << 0;
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/** 如果这个碰撞器是一个触发器,它将不会引起碰撞,但它仍然会触发事件 */
|
2020-06-12 20:24:51 +08:00
|
|
|
|
public isTrigger: boolean;
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 这个对撞机在物理系统注册时的边界。
|
|
|
|
|
|
* 存储这个允许我们始终能够安全地从物理系统中移除对撞机,即使它在试图移除它之前已经被移动了。
|
|
|
|
|
|
*/
|
|
|
|
|
|
public registeredPhysicsBounds: Rectangle = new Rectangle();
|
|
|
|
|
|
/** 如果为true,碰撞器将根据附加的变换缩放和旋转 */
|
|
|
|
|
|
public shouldColliderScaleAndRotateWithTransform = true;
|
|
|
|
|
|
/** 默认为所有层。 */
|
2020-06-16 00:04:28 +08:00
|
|
|
|
public collidesWithLayers = Physics.allLayers;
|
2020-06-15 10:42:06 +08:00
|
|
|
|
|
2020-06-16 00:04:28 +08:00
|
|
|
|
public _localOffsetLength: number;
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/** 标记来跟踪我们的实体是否被添加到场景中 */
|
2020-06-15 10:42:06 +08:00
|
|
|
|
protected _isParentEntityAddedToScene;
|
|
|
|
|
|
protected _colliderRequiresAutoSizing;
|
2020-06-16 11:59:40 +08:00
|
|
|
|
protected _localOffset: Vector2 = new Vector2(0, 0);
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/** 标记来记录我们是否注册了物理系统 */
|
2020-06-16 16:35:17 +08:00
|
|
|
|
protected _isColliderRegistered;
|
2020-06-12 08:47:13 +08:00
|
|
|
|
|
|
|
|
|
|
public get bounds(): Rectangle {
|
2020-07-08 18:12:17 +08:00
|
|
|
|
// this.shape.recalculateBounds(this);
|
|
|
|
|
|
let bds = this.entity.getBounds();
|
|
|
|
|
|
return new Rectangle(bds.x, bds.y, bds.width, bds.height);
|
2020-06-12 08:47:13 +08:00
|
|
|
|
}
|
2020-06-15 10:42:06 +08:00
|
|
|
|
|
2020-06-15 20:08:21 +08:00
|
|
|
|
public get localOffset(){
|
2020-07-08 18:12:17 +08:00
|
|
|
|
return new Vector2(this.x, this.y);
|
2020-06-15 20:08:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 将localOffset添加到实体。获取碰撞器的最终位置。这允许您向一个实体添加多个碰撞器并分别定位它们。
|
|
|
|
|
|
*/
|
2020-06-15 20:08:21 +08:00
|
|
|
|
public set localOffset(value: Vector2){
|
|
|
|
|
|
this.setLocalOffset(value);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public setLocalOffset(offset: Vector2){
|
|
|
|
|
|
if (this._localOffset != offset){
|
|
|
|
|
|
this.unregisterColliderWithPhysicsSystem();
|
2020-07-08 18:12:17 +08:00
|
|
|
|
this.$setX(offset.x);
|
|
|
|
|
|
this.$setY(offset.y);
|
2020-06-16 00:04:28 +08:00
|
|
|
|
this._localOffsetLength = this._localOffset.length();
|
2020-06-15 20:08:21 +08:00
|
|
|
|
this.registerColliderWithPhysicsSystem();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 父实体会在不同的时间调用它(当添加到场景,启用,等等)
|
|
|
|
|
|
*/
|
2020-06-15 20:08:21 +08:00
|
|
|
|
public registerColliderWithPhysicsSystem(){
|
2020-07-08 18:12:17 +08:00
|
|
|
|
// 如果在将我们添加到实体之前更改了origin等属性,则实体可以为null
|
2020-06-16 16:35:17 +08:00
|
|
|
|
if (this._isParentEntityAddedToScene && !this._isColliderRegistered){
|
2020-06-15 20:08:21 +08:00
|
|
|
|
Physics.addCollider(this);
|
2020-06-16 16:35:17 +08:00
|
|
|
|
this._isColliderRegistered = true;
|
2020-06-15 20:08:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 父实体会在不同的时候调用它(从场景中移除,禁用,等等)
|
|
|
|
|
|
*/
|
2020-06-15 20:08:21 +08:00
|
|
|
|
public unregisterColliderWithPhysicsSystem(){
|
2020-06-16 16:35:17 +08:00
|
|
|
|
if (this._isParentEntityAddedToScene && this._isColliderRegistered){
|
2020-06-15 20:08:21 +08:00
|
|
|
|
Physics.removeCollider(this);
|
|
|
|
|
|
}
|
2020-06-16 16:35:17 +08:00
|
|
|
|
this._isColliderRegistered = false;
|
2020-06-15 20:08:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 检查这个形状是否与物理系统中的其他对撞机重叠
|
|
|
|
|
|
* @param other
|
|
|
|
|
|
*/
|
2020-06-16 00:04:28 +08:00
|
|
|
|
public overlaps(other: Collider){
|
|
|
|
|
|
return this.shape.overlaps(other.shape);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-07-08 15:15:15 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 检查这个与运动应用的碰撞器(移动向量)是否与碰撞器碰撞。如果是这样,将返回true,并且结果将填充碰撞数据。
|
|
|
|
|
|
* @param collider
|
|
|
|
|
|
* @param motion
|
|
|
|
|
|
*/
|
2020-06-16 11:22:37 +08:00
|
|
|
|
public collidesWith(collider: Collider, motion: Vector2){
|
2020-07-08 15:15:15 +08:00
|
|
|
|
// 改变形状的位置,使它在移动后的位置,这样我们可以检查重叠
|
2020-06-16 11:22:37 +08:00
|
|
|
|
let oldPosition = this.shape.position;
|
|
|
|
|
|
this.shape.position = Vector2.add(this.shape.position, motion);
|
|
|
|
|
|
|
|
|
|
|
|
let result = this.shape.collidesWithShape(collider.shape);
|
|
|
|
|
|
if (result)
|
|
|
|
|
|
result.collider = collider;
|
|
|
|
|
|
|
2020-07-08 15:15:15 +08:00
|
|
|
|
// 将图形位置返回到检查前的位置
|
2020-06-16 11:22:37 +08:00
|
|
|
|
this.shape.position = oldPosition;
|
|
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public onAddedToEntity(){
|
|
|
|
|
|
if (this._colliderRequiresAutoSizing){
|
|
|
|
|
|
if (!(this instanceof BoxCollider)){
|
|
|
|
|
|
console.error("Only box and circle colliders can be created automatically");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
let bounds = this.entity.getBounds();
|
|
|
|
|
|
let renderbaleBounds = new Rectangle(bounds.x, bounds.y, bounds.width, bounds.height);
|
2020-06-16 11:22:37 +08:00
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
// 这里我们需要大小*反尺度,因为当我们自动调整碰撞器的大小时,它需要没有缩放的渲染
|
|
|
|
|
|
let width = renderbaleBounds.width / this.entity.scale.x;
|
|
|
|
|
|
let height = renderbaleBounds.height / this.entity.scale.y;
|
2020-06-16 11:22:37 +08:00
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
if (this instanceof BoxCollider){
|
|
|
|
|
|
let boxCollider = this as BoxCollider;
|
|
|
|
|
|
boxCollider.width = width;
|
|
|
|
|
|
boxCollider.height = height;
|
2020-06-16 11:22:37 +08:00
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
// 获取渲染的中心,将其转移到本地坐标,并使用它作为碰撞器的localOffset
|
|
|
|
|
|
this.localOffset = Vector2.subtract(renderbaleBounds.center, this.entity.position);
|
2020-06-16 11:22:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
this._isParentEntityAddedToScene = true;
|
|
|
|
|
|
this.registerColliderWithPhysicsSystem();
|
|
|
|
|
|
}
|
2020-06-16 20:22:22 +08:00
|
|
|
|
|
|
|
|
|
|
public onRemovedFromEntity(){
|
|
|
|
|
|
this.unregisterColliderWithPhysicsSystem();
|
|
|
|
|
|
this._isParentEntityAddedToScene = false;
|
|
|
|
|
|
}
|
2020-06-16 11:22:37 +08:00
|
|
|
|
|
2020-06-16 00:04:28 +08:00
|
|
|
|
public onEnabled(){
|
|
|
|
|
|
this.registerColliderWithPhysicsSystem();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public onDisabled(){
|
|
|
|
|
|
this.unregisterColliderWithPhysicsSystem();
|
2020-06-15 10:42:06 +08:00
|
|
|
|
}
|
2020-07-08 15:15:15 +08:00
|
|
|
|
|
2020-07-08 18:12:17 +08:00
|
|
|
|
public onEntityTransformChanged(comp: TransformComponent){
|
|
|
|
|
|
if (this._isColliderRegistered)
|
|
|
|
|
|
Physics.updateCollider(this);
|
2020-07-08 15:15:15 +08:00
|
|
|
|
}
|
2020-06-12 08:47:13 +08:00
|
|
|
|
}
|