2014-08-08 11:16:33 +08:00
|
|
|
/****************************************************************************
|
2018-01-29 16:25:32 +08:00
|
|
|
Copyright (c) 2014-2016 Chukong Technologies Inc.
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2014-08-08 11:16:33 +08:00
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2023-06-11 13:08:08 +08:00
|
|
|
#include "3d/Ray.h"
|
2014-08-08 11:16:33 +08:00
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
NS_AX_BEGIN
|
2014-08-08 11:16:33 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
Ray::Ray() : _direction(0, 0, 1) {}
|
2014-08-08 11:16:33 +08:00
|
|
|
|
2015-08-07 17:36:35 +08:00
|
|
|
Ray::Ray(const Ray& ray)
|
|
|
|
{
|
|
|
|
set(ray._origin, ray._direction);
|
|
|
|
}
|
|
|
|
|
2014-08-08 11:16:33 +08:00
|
|
|
Ray::Ray(const Vec3& origin, const Vec3& direction)
|
|
|
|
{
|
|
|
|
set(origin, direction);
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
Ray::~Ray() {}
|
2014-08-08 11:16:33 +08:00
|
|
|
|
2015-01-16 13:23:03 +08:00
|
|
|
bool Ray::intersects(const AABB& box, float* distance) const
|
2014-08-08 11:16:33 +08:00
|
|
|
{
|
2015-01-16 13:23:03 +08:00
|
|
|
float lowt = 0.0f;
|
2014-08-08 11:16:33 +08:00
|
|
|
float t;
|
2015-01-16 13:23:03 +08:00
|
|
|
bool hit = false;
|
|
|
|
Vec3 hitpoint;
|
2021-12-25 10:04:45 +08:00
|
|
|
const Vec3& min = box._min;
|
|
|
|
const Vec3& max = box._max;
|
2015-01-16 13:23:03 +08:00
|
|
|
const Vec3& rayorig = _origin;
|
2021-12-25 10:04:45 +08:00
|
|
|
const Vec3& raydir = _direction;
|
|
|
|
|
2015-01-16 13:23:03 +08:00
|
|
|
// Check origin inside first
|
|
|
|
if (rayorig > min && rayorig < max)
|
|
|
|
return true;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2015-01-16 13:23:03 +08:00
|
|
|
// Check each face in turn, only check closest 3
|
|
|
|
// Min x
|
|
|
|
if (rayorig.x <= min.x && raydir.x > 0)
|
2014-08-11 10:40:34 +08:00
|
|
|
{
|
2015-01-16 13:23:03 +08:00
|
|
|
t = (min.x - rayorig.x) / raydir.x;
|
|
|
|
if (t >= 0)
|
2014-08-11 10:40:34 +08:00
|
|
|
{
|
2015-01-16 13:23:03 +08:00
|
|
|
// Substitute t back into ray and check bounds and dist
|
|
|
|
hitpoint = rayorig + raydir * t;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (hitpoint.y >= min.y && hitpoint.y <= max.y && hitpoint.z >= min.z && hitpoint.z <= max.z &&
|
2015-01-16 13:23:03 +08:00
|
|
|
(!hit || t < lowt))
|
2014-08-11 10:40:34 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
hit = true;
|
2015-01-16 13:23:03 +08:00
|
|
|
lowt = t;
|
2014-08-11 10:40:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-01-16 13:23:03 +08:00
|
|
|
// Max x
|
|
|
|
if (rayorig.x >= max.x && raydir.x < 0)
|
2014-08-11 10:40:34 +08:00
|
|
|
{
|
2015-01-16 13:23:03 +08:00
|
|
|
t = (max.x - rayorig.x) / raydir.x;
|
|
|
|
if (t >= 0)
|
2014-08-11 10:40:34 +08:00
|
|
|
{
|
2015-01-16 13:23:03 +08:00
|
|
|
// Substitute t back into ray and check bounds and dist
|
|
|
|
hitpoint = rayorig + raydir * t;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (hitpoint.y >= min.y && hitpoint.y <= max.y && hitpoint.z >= min.z && hitpoint.z <= max.z &&
|
2015-01-16 13:23:03 +08:00
|
|
|
(!hit || t < lowt))
|
2014-08-11 10:40:34 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
hit = true;
|
2015-01-16 13:23:03 +08:00
|
|
|
lowt = t;
|
2014-08-11 10:40:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-01-16 13:23:03 +08:00
|
|
|
// Min y
|
|
|
|
if (rayorig.y <= min.y && raydir.y > 0)
|
|
|
|
{
|
|
|
|
t = (min.y - rayorig.y) / raydir.y;
|
|
|
|
if (t >= 0)
|
2014-08-11 10:40:34 +08:00
|
|
|
{
|
2015-01-16 13:23:03 +08:00
|
|
|
// Substitute t back into ray and check bounds and dist
|
|
|
|
hitpoint = rayorig + raydir * t;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (hitpoint.x >= min.x && hitpoint.x <= max.x && hitpoint.z >= min.z && hitpoint.z <= max.z &&
|
2015-01-16 13:23:03 +08:00
|
|
|
(!hit || t < lowt))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
hit = true;
|
2015-01-16 13:23:03 +08:00
|
|
|
lowt = t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Max y
|
|
|
|
if (rayorig.y >= max.y && raydir.y < 0)
|
|
|
|
{
|
|
|
|
t = (max.y - rayorig.y) / raydir.y;
|
|
|
|
if
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2015-01-16 13:23:03 +08:00
|
|
|
(t >= 0)
|
|
|
|
{
|
|
|
|
// Substitute t back into ray and check bounds and dist
|
|
|
|
hitpoint = rayorig + raydir * t;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (hitpoint.x >= min.x && hitpoint.x <= max.x && hitpoint.z >= min.z && hitpoint.z <= max.z &&
|
2015-01-16 13:23:03 +08:00
|
|
|
(!hit || t < lowt))
|
2014-08-11 10:40:34 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
hit = true;
|
2015-01-16 13:23:03 +08:00
|
|
|
lowt = t;
|
2014-08-11 10:40:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-01-16 13:23:03 +08:00
|
|
|
// Min z
|
|
|
|
if (rayorig.z <= min.z && raydir.z > 0)
|
|
|
|
{
|
|
|
|
t = (min.z - rayorig.z) / raydir.z;
|
|
|
|
if (t >= 0)
|
|
|
|
{
|
|
|
|
// Substitute t back into ray and check bounds and dist
|
|
|
|
hitpoint = rayorig + raydir * t;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (hitpoint.x >= min.x && hitpoint.x <= max.x && hitpoint.y >= min.y && hitpoint.y <= max.y &&
|
2015-01-16 13:23:03 +08:00
|
|
|
(!hit || t < lowt))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
hit = true;
|
2015-01-16 13:23:03 +08:00
|
|
|
lowt = t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Max z
|
|
|
|
if (rayorig.z >= max.z && raydir.z < 0)
|
|
|
|
{
|
|
|
|
t = (max.z - rayorig.z) / raydir.z;
|
|
|
|
if (t >= 0)
|
|
|
|
{
|
|
|
|
// Substitute t back into ray and check bounds and dist
|
|
|
|
hitpoint = rayorig + raydir * t;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (hitpoint.x >= min.x && hitpoint.x <= max.x && hitpoint.y >= min.y && hitpoint.y <= max.y &&
|
2015-01-16 13:23:03 +08:00
|
|
|
(!hit || t < lowt))
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
hit = true;
|
2015-01-16 13:23:03 +08:00
|
|
|
lowt = t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2015-01-16 13:23:03 +08:00
|
|
|
if (distance)
|
|
|
|
*distance = lowt;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2015-01-16 13:23:03 +08:00
|
|
|
return hit;
|
2014-08-08 11:16:33 +08:00
|
|
|
}
|
|
|
|
|
2015-01-16 13:23:03 +08:00
|
|
|
bool Ray::intersects(const OBB& obb, float* distance) const
|
2014-08-08 11:16:33 +08:00
|
|
|
{
|
|
|
|
AABB aabb;
|
2021-12-25 10:04:45 +08:00
|
|
|
aabb._min = -obb._extents;
|
2014-08-08 11:16:33 +08:00
|
|
|
aabb._max = obb._extents;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2014-08-08 11:16:33 +08:00
|
|
|
Ray ray;
|
|
|
|
ray._direction = _direction;
|
2021-12-25 10:04:45 +08:00
|
|
|
ray._origin = _origin;
|
|
|
|
|
2014-08-08 11:16:33 +08:00
|
|
|
Mat4 mat = Mat4::IDENTITY;
|
|
|
|
mat.m[0] = obb._xAxis.x;
|
|
|
|
mat.m[1] = obb._xAxis.y;
|
|
|
|
mat.m[2] = obb._xAxis.z;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2014-08-08 11:16:33 +08:00
|
|
|
mat.m[4] = obb._yAxis.x;
|
|
|
|
mat.m[5] = obb._yAxis.y;
|
|
|
|
mat.m[6] = obb._yAxis.z;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
mat.m[8] = obb._zAxis.x;
|
|
|
|
mat.m[9] = obb._zAxis.y;
|
2014-08-08 11:16:33 +08:00
|
|
|
mat.m[10] = obb._zAxis.z;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2014-08-08 11:16:33 +08:00
|
|
|
mat.m[12] = obb._center.x;
|
|
|
|
mat.m[13] = obb._center.y;
|
|
|
|
mat.m[14] = obb._center.z;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2014-08-08 11:16:33 +08:00
|
|
|
mat = mat.getInversed();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2014-08-08 11:16:33 +08:00
|
|
|
ray.transform(mat);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2015-01-16 13:23:03 +08:00
|
|
|
return ray.intersects(aabb, distance);
|
2014-08-08 11:16:33 +08:00
|
|
|
}
|
|
|
|
|
2014-12-16 16:34:35 +08:00
|
|
|
float Ray::dist(const Plane& plane) const
|
|
|
|
{
|
2014-12-22 18:13:07 +08:00
|
|
|
float ndd = Vec3::dot(plane.getNormal(), _direction);
|
2021-12-25 10:04:45 +08:00
|
|
|
if (ndd == 0)
|
2014-12-22 18:13:07 +08:00
|
|
|
return 0.0f;
|
|
|
|
float ndo = Vec3::dot(plane.getNormal(), _origin);
|
|
|
|
return (plane.getDist() - ndo) / ndd;
|
2014-12-16 16:34:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Vec3 Ray::intersects(const Plane& plane) const
|
|
|
|
{
|
|
|
|
float dis = this->dist(plane);
|
2014-12-22 18:13:07 +08:00
|
|
|
return _origin + dis * _direction;
|
2014-12-16 16:34:35 +08:00
|
|
|
}
|
|
|
|
|
2014-08-08 11:16:33 +08:00
|
|
|
void Ray::set(const Vec3& origin, const Vec3& direction)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
_origin = origin;
|
2014-08-08 11:16:33 +08:00
|
|
|
_direction = direction;
|
2014-12-16 16:34:35 +08:00
|
|
|
_direction.normalize();
|
2014-08-08 11:16:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Ray::transform(const Mat4& matrix)
|
|
|
|
{
|
|
|
|
matrix.transformPoint(&_origin);
|
|
|
|
matrix.transformVector(&_direction);
|
|
|
|
_direction.normalize();
|
|
|
|
}
|
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
NS_AX_END
|