/** * Copyright 2008 - 2015 The Loon Game Engine Authors * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. * * @project loon * @author cping * @email:javachenpeng@yahoo.com * @version 0.5 */ package loon.geom; import loon.geom.Matrix4; import loon.geom.Vector3f; import loon.utils.MathUtils; import loon.utils.TArray; public class BoundingBox implements XY { private final static Vector3f tmpVector = new Vector3f(); public final Vector3f min = new Vector3f(); public final Vector3f max = new Vector3f(); private final Vector3f cnt = new Vector3f(); private final Vector3f dim = new Vector3f(); private Vector3f[] corners; public Vector3f getCenter() { return cnt; } public Vector3f getCenter(Vector3f out) { return out.set(cnt); } public float getCenterX() { return cnt.x; } public float getCenterY() { return cnt.y; } public float getCenterZ() { return cnt.z; } public Vector3f[] getCorners() { if (corners == null) { corners = new Vector3f[8]; for (int i = 0; i < 8; i++) corners[i] = new Vector3f(); } corners[0].set(min.x, min.y, min.z); corners[1].set(max.x, min.y, min.z); corners[2].set(max.x, max.y, min.z); corners[3].set(min.x, max.y, min.z); corners[4].set(min.x, min.y, max.z); corners[5].set(max.x, min.y, max.z); corners[6].set(max.x, max.y, max.z); corners[7].set(min.x, max.y, max.z); return corners; } public Vector3f getCorner000(final Vector3f out) { return out.set(min.x, min.y, min.z); } public Vector3f getCorner001(final Vector3f out) { return out.set(min.x, min.y, max.z); } public Vector3f getCorner010(final Vector3f out) { return out.set(min.x, max.y, min.z); } public Vector3f getCorner011(final Vector3f out) { return out.set(min.x, max.y, max.z); } public Vector3f getCorner100(final Vector3f out) { return out.set(max.x, min.y, min.z); } public Vector3f getCorner101(final Vector3f out) { return out.set(max.x, min.y, max.z); } public Vector3f getCorner110(final Vector3f out) { return out.set(max.x, max.y, min.z); } public Vector3f getCorner111(final Vector3f out) { return out.set(max.x, max.y, max.z); } public Vector3f getDimensions() { return dim; } public Vector3f getDimensions(final Vector3f out) { return out.set(dim); } @Override public float getX() { return min.x; } @Override public float getY() { return min.y; } public float getWidth() { return dim.x; } public float getHeight() { return dim.y; } public float getDepth() { return dim.z; } public Vector3f getMin(final Vector3f out) { return out.set(min); } public Vector3f getMax(final Vector3f out) { return out.set(max); } public BoundingBox() { clr(); } public BoundingBox(BoundingBox bounds) { this.set(bounds); } public BoundingBox(Vector3f minimum, Vector3f maximum) { this.set(minimum, maximum); } public BoundingBox set(BoundingBox bounds) { return this.set(bounds.min, bounds.max); } public BoundingBox set(Vector3f minimum, Vector3f maximum) { min.set(minimum.x < maximum.x ? minimum.x : maximum.x, minimum.y < maximum.y ? minimum.y : maximum.y, minimum.z < maximum.z ? minimum.z : maximum.z); max.set(minimum.x > maximum.x ? minimum.x : maximum.x, minimum.y > maximum.y ? minimum.y : maximum.y, minimum.z > maximum.z ? minimum.z : maximum.z); cnt.set(min).addSelf(max).scaleSelf(0.5f); dim.set(max).subtractSelf(min); return this; } public BoundingBox set(Vector3f[] points) { this.inf(); for (Vector3f l_point : points) this.ext(l_point); return this; } public BoundingBox set(TArray<Vector3f> points) { this.inf(); for (Vector3f l_point : points) this.ext(l_point); return this; } public BoundingBox inf() { min.set(Float.POSITIVE_INFINITY, Float.POSITIVE_INFINITY, Float.POSITIVE_INFINITY); max.set(Float.NEGATIVE_INFINITY, Float.NEGATIVE_INFINITY, Float.NEGATIVE_INFINITY); cnt.set(0, 0, 0); dim.set(0, 0, 0); return this; } public BoundingBox ext(Vector3f point) { return this.set( min.set(min(min.x, point.x), min(min.y, point.y), min(min.z, point.z)), max.set(MathUtils.max(max.x, point.x), MathUtils.max(max.y, point.y), MathUtils.max(max.z, point.z))); } public BoundingBox clr() { return this.set(min.set(0, 0, 0), max.set(0, 0, 0)); } public boolean isValid() { return min.x < max.x && min.y < max.y && min.z < max.z; } public BoundingBox ext(BoundingBox a_bounds) { return this.set(min.set(min(min.x, a_bounds.min.x), min(min.y, a_bounds.min.y), min(min.z, a_bounds.min.z)), max .set(max(max.x, a_bounds.max.x), max(max.y, a_bounds.max.y), max(max.z, a_bounds.max.z))); } public BoundingBox ext(BoundingBox bounds, Matrix4 transform) { ext(tmpVector.set(bounds.min.x, bounds.min.y, bounds.min.z).mulSelf( transform)); ext(tmpVector.set(bounds.min.x, bounds.min.y, bounds.max.z).mulSelf( transform)); ext(tmpVector.set(bounds.min.x, bounds.max.y, bounds.min.z).mulSelf( transform)); ext(tmpVector.set(bounds.min.x, bounds.max.y, bounds.max.z).mulSelf( transform)); ext(tmpVector.set(bounds.max.x, bounds.min.y, bounds.min.z).mulSelf( transform)); ext(tmpVector.set(bounds.max.x, bounds.min.y, bounds.max.z).mulSelf( transform)); ext(tmpVector.set(bounds.max.x, bounds.max.y, bounds.min.z).mulSelf( transform)); ext(tmpVector.set(bounds.max.x, bounds.max.y, bounds.max.z).mulSelf( transform)); return this; } public BoundingBox mulSelf(Matrix4 transform) { final float x0 = min.x, y0 = min.y, z0 = min.z, x1 = max.x, y1 = max.y, z1 = max.z; inf(); ext(tmpVector.set(x0, y0, z0).mulSelf(transform)); ext(tmpVector.set(x0, y0, z1).mulSelf(transform)); ext(tmpVector.set(x0, y1, z0).mulSelf(transform)); ext(tmpVector.set(x0, y1, z1).mulSelf(transform)); ext(tmpVector.set(x1, y0, z0).mulSelf(transform)); ext(tmpVector.set(x1, y0, z1).mulSelf(transform)); ext(tmpVector.set(x1, y1, z0).mulSelf(transform)); ext(tmpVector.set(x1, y1, z1).mulSelf(transform)); return this; } public boolean contains(BoundingBox b) { return !isValid() || (min.x <= b.min.x && min.y <= b.min.y && min.z <= b.min.z && max.x >= b.max.x && max.y >= b.max.y && max.z >= b.max.z); } public boolean intersects(BoundingBox b) { if (!isValid()) { return false; } float lx = MathUtils.abs(this.cnt.x - b.cnt.x); float sumx = (this.dim.x / 2.0f) + (b.dim.x / 2.0f); float ly = MathUtils.abs(this.cnt.y - b.cnt.y); float sumy = (this.dim.y / 2.0f) + (b.dim.y / 2.0f); float lz = MathUtils.abs(this.cnt.z - b.cnt.z); float sumz = (this.dim.z / 2.0f) + (b.dim.z / 2.0f); return (lx <= sumx && ly <= sumy && lz <= sumz); } public boolean contains(Vector3f v) { return min.x <= v.x && max.x >= v.x && min.y <= v.y && max.y >= v.y && min.z <= v.z && max.z >= v.z; } @Override public String toString() { return "[" + min + "|" + max + "]"; } public BoundingBox ext(float x, float y, float z) { return this.set(min.set(min(min.x, x), min(min.y, y), min(min.z, z)), max.set(max(max.x, x), max(max.y, y), max(max.z, z))); } static final float min(final float a, final float b) { return a > b ? b : a; } static final float max(final float a, final float b) { return a > b ? a : b; } }