Fixed raycast sphere.

This commit is contained in:
2023-03-18 22:58:26 -07:00
parent 669b02ca06
commit f8ab4e2abc
9 changed files with 249 additions and 189 deletions

View File

@ -131,7 +131,7 @@ void Transform::setLocalTransform(glm::mat4 transform) {
} }
glm::vec3 Transform::getWorldPosition() { glm::vec3 Transform::getWorldPosition() {
return glm::inverse(this->transformWorld)[3]; return this->transformWorld[3];
} }
glm::mat4 Transform::getWorldTransform() { glm::mat4 Transform::getWorldTransform() {

View File

@ -14,20 +14,23 @@ bool_t Dawn::raytestSphere(
glm::vec3 *normal, glm::vec3 *normal,
float_t *distance float_t *distance
) { ) {
assertNotNull(hit); float_t a = glm::dot(ray.direction, ray.direction);
assertNotNull(normal); float_t b = 2.0f * glm::dot(ray.direction, ray.origin - sphere.center);
float_t c = glm::dot(ray.origin - sphere.center, ray.origin - sphere.center);
c -= sphere.radius * sphere.radius;
glm::vec3 h, n; float_t dt = b * b - 4.0f * a * c;
auto result = glm::intersectRaySphere( if(dt < 0.0f) return false;
ray.origin, ray.direction,
sphere.center, sphere.radius, float_t t0 = (-b - sqrtf(dt)) / (a * 2.0f);
h, n if(t0 < 0.0f) return false;
);
if(!result) return result; *hit = ray.origin + t0 * ray.direction;
*hit = h; *normal = glm::normalize(*hit - sphere.center);
*normal = n; *distance = t0;
*distance = glm::distance(ray.origin, h);
return result;
return true;
} }
bool_t Dawn::raytestTriangle( bool_t Dawn::raytestTriangle(

View File

@ -9,4 +9,5 @@ target_sources(${DAWN_TARGET_NAME}
Collider3D.cpp Collider3D.cpp
CubeCollider.cpp CubeCollider.cpp
CapsuleCollider.cpp CapsuleCollider.cpp
SphereCollider.cpp
) )

View File

@ -17,7 +17,8 @@ namespace Dawn {
enum Collider3DType { enum Collider3DType {
COLLIDER3D_TYPE_CUBE, COLLIDER3D_TYPE_CUBE,
COLLIDER3D_TYPE_CAPSULE COLLIDER3D_TYPE_CAPSULE,
COLLIDER3D_TYPE_SPHERE
}; };
class Collider3D : public SceneItemComponent { class Collider3D : public SceneItemComponent {

View File

@ -0,0 +1,31 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "SphereCollider.hpp"
using namespace Dawn;
SphereCollider::SphereCollider(SceneItem *item) : Collider3D(item) {
}
enum Collider3DType SphereCollider::getColliderType() {
return COLLIDER3D_TYPE_SPHERE;
}
bool_t SphereCollider::performRaycast(
struct Collider3DRayResult *result,
struct Ray3D ray
) {
assertNotNull(result);
return raytestSphere(
ray,
{ .center = transform->getLocalPosition(), .radius = this->radius },
&result->point,
&result->normal,
&result->distance
);
}

View File

@ -0,0 +1,24 @@
// Copyright (c) 2023 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#pragma once
#include "Collider3D.hpp"
namespace Dawn {
class SphereCollider : public Collider3D {
protected:
bool_t performRaycast(
struct Collider3DRayResult *result,
struct Ray3D ray
) override;
public:
float_t radius = 1.0f;
SphereCollider(SceneItem *item);
enum Collider3DType getColliderType() override;
};
}

View File

@ -1,165 +1,165 @@
// Copyright (c) 2023 Dominic Masters // Copyright (c) 2023 Dominic Masters
// //
// This software is released under the MIT License. // This software is released under the MIT License.
// https://opensource.org/licenses/MIT // https://opensource.org/licenses/MIT
#include "SceneDebugLine.hpp" #include "SceneDebugLine.hpp"
#include "display/shader/SimpleTexturedShader.hpp" #include "display/shader/SimpleTexturedShader.hpp"
#include "scene/components/display/Camera.hpp" #include "scene/components/display/Camera.hpp"
#include "scene/Scene.hpp" #include "scene/Scene.hpp"
#include "scene/components/physics/3d/Collider3D.hpp" #include "scene/components/physics/3d/Collider3D.hpp"
#include "scene/components/physics/3d/CubeCollider.hpp" #include "scene/components/physics/3d/CubeCollider.hpp"
using namespace Dawn; using namespace Dawn;
struct ShaderPassItem SceneDebugLine::createShaderItem( struct ShaderPassItem SceneDebugLine::createShaderItem(
Mesh *mesh, Mesh *mesh,
int32_t *lineIndex, int32_t *lineIndex,
Camera *camera, Camera *camera,
SimpleTexturedShader *shader SimpleTexturedShader *shader
) { ) {
struct ShaderPassItem item; struct ShaderPassItem item;
item.priority = this->priority; item.priority = this->priority;
item.shaderProgram = &shader->program; item.shaderProgram = &shader->program;
item.colorValues[shader->program.paramColor] = this->color; item.colorValues[shader->program.paramColor] = this->color;
item.matrixValues[shader->program.paramModel] = this->transform; item.matrixValues[shader->program.paramModel] = this->transform;
item.matrixValues[shader->program.paramView] = camera->transform->getWorldTransform(); item.matrixValues[shader->program.paramView] = camera->transform->getWorldTransform();
item.matrixValues[shader->program.paramProjection] = camera->getProjection(); item.matrixValues[shader->program.paramProjection] = camera->getProjection();
item.boolValues[shader->program.paramHasTexture] = false; item.boolValues[shader->program.paramHasTexture] = false;
auto i = *lineIndex; auto i = *lineIndex;
item.mesh = mesh; item.mesh = mesh;
item.drawMode = MESH_DRAW_MODE_LINES; item.drawMode = MESH_DRAW_MODE_LINES;
item.renderFlags = 0x00; item.renderFlags = RENDER_MANAGER_RENDER_FLAG_DEPTH_TEST;
item.start = i * SCENE_DEBUG_LINE_INDICE_COUNT; item.start = i * SCENE_DEBUG_LINE_INDICE_COUNT;
item.count = SCENE_DEBUG_LINE_INDICE_COUNT; item.count = SCENE_DEBUG_LINE_INDICE_COUNT;
glm::vec3 positions[SCENE_DEBUG_LINE_VERTICE_COUNT] = { this->v0, this->v1 }; glm::vec3 positions[SCENE_DEBUG_LINE_VERTICE_COUNT] = { this->v0, this->v1 };
mesh->bufferPositions( mesh->bufferPositions(
i * SCENE_DEBUG_LINE_VERTICE_COUNT, i * SCENE_DEBUG_LINE_VERTICE_COUNT,
positions, positions,
SCENE_DEBUG_LINE_VERTICE_COUNT SCENE_DEBUG_LINE_VERTICE_COUNT
); );
meshindice_t indices[SCENE_DEBUG_LINE_INDICE_COUNT] = { meshindice_t indices[SCENE_DEBUG_LINE_INDICE_COUNT] = {
i * SCENE_DEBUG_LINE_VERTICE_COUNT, i * SCENE_DEBUG_LINE_VERTICE_COUNT,
(i*SCENE_DEBUG_LINE_VERTICE_COUNT) + 1 (i*SCENE_DEBUG_LINE_VERTICE_COUNT) + 1
}; };
mesh->bufferIndices( mesh->bufferIndices(
i * SCENE_DEBUG_LINE_INDICE_COUNT, i * SCENE_DEBUG_LINE_INDICE_COUNT,
indices, indices,
SCENE_DEBUG_LINE_INDICE_COUNT SCENE_DEBUG_LINE_INDICE_COUNT
); );
return item; return item;
} }
// Scene Implementations (done here to keep things cleaner in scene) // Scene Implementations (done here to keep things cleaner in scene)
void Scene::debugLine(struct SceneDebugLine line) { void Scene::debugLine(struct SceneDebugLine line) {
this->debugLines.push_back(line); this->debugLines.push_back(line);
} }
void Scene::debugRay(struct SceneDebugRay ray) { void Scene::debugRay(struct SceneDebugRay ray) {
this->debugLine((struct SceneDebugLine){ this->debugLine((struct SceneDebugLine){
.v0 = ray.start, .v0 = ray.start,
.v1 = ray.start + ray.direction, .v1 = ray.start + ray.direction,
.color = ray.color .color = ray.color
}); });
} }
void Scene::debugCube(struct SceneDebugCube cube) { void Scene::debugCube(struct SceneDebugCube cube) {
auto min = cube.min; auto min = cube.min;
auto max = cube.max; auto max = cube.max;
struct SceneDebugLine line; struct SceneDebugLine line;
line.color = cube.color; line.color = cube.color;
line.transform = cube.transform; line.transform = cube.transform;
// Bottom Face // Bottom Face
line.v0 = glm::vec3(min.x, min.y, min.z), line.v1 = glm::vec3(max.x, min.y, min.z); line.v0 = glm::vec3(min.x, min.y, min.z), line.v1 = glm::vec3(max.x, min.y, min.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(min.x, min.y, min.z), line.v1 = glm::vec3(min.x, min.y, max.z); line.v0 = glm::vec3(min.x, min.y, min.z), line.v1 = glm::vec3(min.x, min.y, max.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(max.x, min.y, min.z), line.v1 = glm::vec3(max.x, min.y, max.z); line.v0 = glm::vec3(max.x, min.y, min.z), line.v1 = glm::vec3(max.x, min.y, max.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(min.x, min.y, max.z), line.v1 = glm::vec3(max.x, min.y, max.z); line.v0 = glm::vec3(min.x, min.y, max.z), line.v1 = glm::vec3(max.x, min.y, max.z);
this->debugLine(line); this->debugLine(line);
// Top Face // Top Face
line.v0 = glm::vec3(min.x, max.y, min.z), line.v1 = glm::vec3(max.x, max.y, min.z); line.v0 = glm::vec3(min.x, max.y, min.z), line.v1 = glm::vec3(max.x, max.y, min.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(min.x, max.y, min.z), line.v1 = glm::vec3(min.x, max.y, max.z); line.v0 = glm::vec3(min.x, max.y, min.z), line.v1 = glm::vec3(min.x, max.y, max.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(max.x, max.y, min.z), line.v1 = glm::vec3(max.x, max.y, max.z); line.v0 = glm::vec3(max.x, max.y, min.z), line.v1 = glm::vec3(max.x, max.y, max.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(min.x, max.y, max.z), line.v1 = glm::vec3(max.x, max.y, max.z); line.v0 = glm::vec3(min.x, max.y, max.z), line.v1 = glm::vec3(max.x, max.y, max.z);
this->debugLine(line); this->debugLine(line);
// Rails // Rails
line.v0 = glm::vec3(min.x, min.y, min.z), line.v1 = glm::vec3(min.x, max.y, min.z); line.v0 = glm::vec3(min.x, min.y, min.z), line.v1 = glm::vec3(min.x, max.y, min.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(max.x, min.y, min.z), line.v1 = glm::vec3(max.x, max.y, min.z); line.v0 = glm::vec3(max.x, min.y, min.z), line.v1 = glm::vec3(max.x, max.y, min.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(min.x, min.y, max.z), line.v1 = glm::vec3(min.x, max.y, max.z); line.v0 = glm::vec3(min.x, min.y, max.z), line.v1 = glm::vec3(min.x, max.y, max.z);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(max.x, min.y, max.z), line.v1 = glm::vec3(max.x, max.y, max.z); line.v0 = glm::vec3(max.x, min.y, max.z), line.v1 = glm::vec3(max.x, max.y, max.z);
this->debugLine(line); this->debugLine(line);
} }
void Scene::debugOrigin() { void Scene::debugOrigin() {
struct SceneDebugLine line; struct SceneDebugLine line;
line.v1 = glm::vec3(1, 0, 0), line.color = COLOR_RED; line.v1 = glm::vec3(1, 0, 0), line.color = COLOR_RED;
this->debugLine(line); this->debugLine(line);
line.v1 = glm::vec3(0, 1, 0), line.color = COLOR_GREEN; line.v1 = glm::vec3(0, 1, 0), line.color = COLOR_GREEN;
this->debugLine(line); this->debugLine(line);
line.v1 = glm::vec3(0, 0, 1), line.color = COLOR_BLUE; line.v1 = glm::vec3(0, 0, 1), line.color = COLOR_BLUE;
this->debugLine(line); this->debugLine(line);
} }
void Scene::debugGrid() { void Scene::debugGrid() {
float_t s = 10.0f; float_t s = 10.0f;
float_t t = 1.0f; float_t t = 1.0f;
struct SceneDebugLine line; struct SceneDebugLine line;
line.color = COLOR_LIGHT_GREY; line.color = COLOR_LIGHT_GREY;
line.color.a = 0.2f; line.color.a = 0.2f;
line.v0 = glm::vec3(-s, 0, -s), line.v1 = glm::vec3(s, 0, -s); line.v0 = glm::vec3(-s, 0, -s), line.v1 = glm::vec3(s, 0, -s);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(-s, 0, -s), line.v1 = glm::vec3(-s, 0, s); line.v0 = glm::vec3(-s, 0, -s), line.v1 = glm::vec3(-s, 0, s);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(s, 0, -s), line.v1 = glm::vec3(s, 0, s); line.v0 = glm::vec3(s, 0, -s), line.v1 = glm::vec3(s, 0, s);
this->debugLine(line); this->debugLine(line);
line.v0 = glm::vec3(-s, 0, s), line.v1 = glm::vec3(s, 0, s); line.v0 = glm::vec3(-s, 0, s), line.v1 = glm::vec3(s, 0, s);
this->debugLine(line); this->debugLine(line);
for(float_t x = -s+t; x < s; x += t) { for(float_t x = -s+t; x < s; x += t) {
line.v0 = glm::vec3(x, 0, -s), line.v1 = glm::vec3(x, 0, s); line.v0 = glm::vec3(x, 0, -s), line.v1 = glm::vec3(x, 0, s);
this->debugLine(line); this->debugLine(line);
} }
for(float_t z = -s+t; z < s; z += t) { for(float_t z = -s+t; z < s; z += t) {
line.v0 = glm::vec3(-s, 0, z), line.v1 = glm::vec3(s, 0, z); line.v0 = glm::vec3(-s, 0, z), line.v1 = glm::vec3(s, 0, z);
this->debugLine(line); this->debugLine(line);
} }
} }
void Scene::debugHitboxes() { void Scene::debugHitboxes() {
auto colliders = this->findComponents<Collider3D>(); auto colliders = this->findComponents<Collider3D>();
auto itColliders = colliders.begin(); auto itColliders = colliders.begin();
while(itColliders != colliders.end()) { while(itColliders != colliders.end()) {
auto c = *itColliders; auto c = *itColliders;
switch(c->getColliderType()) { switch(c->getColliderType()) {
case COLLIDER3D_TYPE_CUBE: case COLLIDER3D_TYPE_CUBE:
auto asCube = dynamic_cast<CubeCollider*>(c); auto asCube = dynamic_cast<CubeCollider*>(c);
this->debugCube((struct SceneDebugCube){ this->debugCube((struct SceneDebugCube){
.min = asCube->min, .min = asCube->min,
.max = asCube->max, .max = asCube->max,
.color = COLOR_BLUE, .color = COLOR_BLUE,
.transform = asCube->transform->getWorldTransform() .transform = asCube->transform->getWorldTransform()
}); });
break; break;
} }
++itColliders; ++itColliders;
} }
} }

View File

@ -39,12 +39,13 @@ void PlayerController::onStart() {
// tEST // tEST
auto collider = item->getComponent<CapsuleCollider>(); auto collider = item->getComponent<Collider3D>();
assertNotNull(collider);
Collider3DRayResult result; Collider3DRayResult result;
struct Ray3D ray; struct Ray3D ray;
ray.origin = glm::vec3(0, 0.5f, 1); ray.origin = glm::vec3(0, 0.0f, 5);
ray.direction = glm::vec3(10, 0, 0); ray.direction = glm::vec3(0, 0.0f, -4);
struct Color color = COLOR_MAGENTA; struct Color color = COLOR_MAGENTA;
@ -54,7 +55,7 @@ void PlayerController::onStart() {
getScene()->debugRay({ getScene()->debugRay({
.start = result.point, .start = result.point,
.direction = (result.normal * 100.0f), .direction = result.normal,
.color = COLOR_GREEN .color = COLOR_GREEN
}); });
} }

View File

@ -7,14 +7,14 @@
#include "scene/Scene.hpp" #include "scene/Scene.hpp"
#include "scene/components/PlayerController.hpp" #include "scene/components/PlayerController.hpp"
#include "scene/components/GameCamera.hpp" #include "scene/components/GameCamera.hpp"
#include "scene/components/physics/3d/CubeCollider.hpp"
#include "scene/components/physics/3d/CapsuleCollider.hpp"
#include "scene/components/GameCamera.hpp" #include "scene/components/GameCamera.hpp"
#include "scene/components/display/MeshRenderer.hpp" #include "scene/components/display/MeshRenderer.hpp"
#include "scene/components/display/MeshHost.hpp" #include "scene/components/display/MeshHost.hpp"
#include "scene/components/display/material/SimpleTexturedMaterial.hpp" #include "scene/components/display/material/SimpleTexturedMaterial.hpp"
#include "display/mesh/CapsuleMesh.hpp"
#include "scene/components/example/ExampleSpin.hpp" #include "scene/components/example/ExampleSpin.hpp"
#include "scene/components/physics/3d/SphereCollider.hpp"
#include "scene/components/physics/3d/CubeCollider.hpp"
#include "display/mesh/SphereMesh.hpp"
namespace Dawn { namespace Dawn {
class HelloWorldScene : public Scene { class HelloWorldScene : public Scene {
@ -25,12 +25,11 @@ namespace Dawn {
auto playerItem = this->createSceneItem(); auto playerItem = this->createSceneItem();
auto player = playerItem->addComponent<PlayerController>(); auto player = playerItem->addComponent<PlayerController>();
auto hitbox = playerItem->addComponent<CapsuleCollider>();
playerItem->addComponent<MeshRenderer>(); playerItem->addComponent<MeshRenderer>();
auto meshHost = playerItem->addComponent<MeshHost>(); auto meshHost = playerItem->addComponent<MeshHost>();
playerItem->addComponent<SimpleTexturedMaterial>(); playerItem->addComponent<SimpleTexturedMaterial>();
auto hitbox = playerItem->addComponent<SphereCollider>();
CapsuleMesh::create(&meshHost->mesh, hitbox->radius, hitbox->height); SphereMesh::createSphere(&meshHost->mesh, hitbox->radius, 32, 32);
auto wall = this->createSceneItem(); auto wall = this->createSceneItem();
auto wallHitbox = wall->addComponent<CubeCollider>(); auto wallHitbox = wall->addComponent<CubeCollider>();