mirror of
https://github.com/Relintai/godot_voxel.git
synced 2025-01-12 20:01:18 +01:00
Added simple voxel raycast
This commit is contained in:
parent
353b32c49a
commit
ef4d07ca03
141
voxel_raycast.cpp
Normal file
141
voxel_raycast.cpp
Normal file
@ -0,0 +1,141 @@
|
||||
#include "voxel_raycast.h"
|
||||
#include <math_funcs.h>
|
||||
|
||||
const float g_infinite = 9999999;
|
||||
|
||||
bool voxel_raycast(
|
||||
Vector3 ray_origin,
|
||||
Vector3 ray_direction,
|
||||
VoxelPredicate predicate,
|
||||
void * predicate_context,
|
||||
real_t max_distance,
|
||||
Vector3i & out_hit_pos,
|
||||
Vector3i & out_prev_pos
|
||||
){
|
||||
// Equation : p + v*t
|
||||
// p : ray start position (ray.pos)
|
||||
// v : ray orientation vector (ray.dir)
|
||||
// t : parametric variable = a distance if v is normalized
|
||||
|
||||
// This raycasting technique is described here :
|
||||
// http://www.cse.yorku.ca/~amana/research/grid.pdf
|
||||
|
||||
// Note : the grid is assumed to have 1-unit square cells.
|
||||
|
||||
ERR_FAIL_COND_V(predicate == 0, false);
|
||||
ERR_FAIL_COND_V(ray_direction.is_normalized() == false, false); // Must be normalized
|
||||
|
||||
/* Initialisation */
|
||||
|
||||
// Voxel position
|
||||
Vector3i hit_pos(
|
||||
Math::floor(ray_origin.x),
|
||||
Math::floor(ray_origin.y),
|
||||
Math::floor(ray_origin.z)
|
||||
);
|
||||
Vector3i hit_prev_pos = hit_pos;
|
||||
|
||||
// Voxel step
|
||||
const int xi_step = ray_direction.x > 0 ? 1 : ray_direction.x < 0 ? -1 : 0;
|
||||
const int yi_step = ray_direction.y > 0 ? 1 : ray_direction.y < 0 ? -1 : 0;
|
||||
const int zi_step = ray_direction.z > 0 ? 1 : ray_direction.z < 0 ? -1 : 0;
|
||||
|
||||
// Parametric voxel step
|
||||
const real_t tdelta_x = xi_step != 0 ? 1.f / Math::abs(ray_direction.x) : g_infinite;
|
||||
const real_t tdelta_y = yi_step != 0 ? 1.f / Math::abs(ray_direction.y) : g_infinite;
|
||||
const real_t tdelta_z = zi_step != 0 ? 1.f / Math::abs(ray_direction.z) : g_infinite;
|
||||
|
||||
// Parametric grid-cross
|
||||
real_t tcross_x; // At which value of T we will cross a vertical line?
|
||||
real_t tcross_y; // At which value of T we will cross a horizontal line?
|
||||
real_t tcross_z; // At which value of T we will cross a depth line?
|
||||
|
||||
// X initialization
|
||||
if(xi_step != 0)
|
||||
{
|
||||
if(xi_step == 1)
|
||||
tcross_x = (Math::ceil(ray_origin.x) - ray_origin.x) * tdelta_x;
|
||||
else
|
||||
tcross_x = (ray_origin.x - Math::floor(ray_origin.x)) * tdelta_x;
|
||||
}
|
||||
else
|
||||
tcross_x = g_infinite; // Will never cross on X
|
||||
|
||||
// Y initialization
|
||||
if(yi_step != 0)
|
||||
{
|
||||
if(yi_step == 1)
|
||||
tcross_y = (Math::ceil(ray_origin.y) - ray_origin.y) * tdelta_y;
|
||||
else
|
||||
tcross_y = (ray_origin.y - Math::floor(ray_origin.y)) * tdelta_y;
|
||||
}
|
||||
else
|
||||
tcross_y = g_infinite; // Will never cross on X
|
||||
|
||||
// Z initialization
|
||||
if(zi_step != 0)
|
||||
{
|
||||
if(zi_step == 1)
|
||||
tcross_z = (Math::ceil(ray_origin.z) - ray_origin.z) * tdelta_z;
|
||||
else
|
||||
tcross_z = (ray_origin.z - Math::floor(ray_origin.z)) * tdelta_z;
|
||||
}
|
||||
else
|
||||
tcross_z = g_infinite; // Will never cross on X
|
||||
|
||||
/* Iteration */
|
||||
|
||||
do
|
||||
{
|
||||
hit_prev_pos = hit_pos;
|
||||
if(tcross_x < tcross_y)
|
||||
{
|
||||
if(tcross_x < tcross_z)
|
||||
{
|
||||
// X collision
|
||||
//hit.prevPos.x = hit.pos.x;
|
||||
hit_pos.x += xi_step;
|
||||
if(tcross_x > max_distance)
|
||||
return false;
|
||||
tcross_x += tdelta_x;
|
||||
}
|
||||
else
|
||||
{
|
||||
// Z collision (duplicate code)
|
||||
//hit.prevPos.z = hit.pos.z;
|
||||
hit_pos.z += zi_step;
|
||||
if(tcross_z > max_distance)
|
||||
return false;
|
||||
tcross_z += tdelta_z;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if(tcross_y < tcross_z)
|
||||
{
|
||||
// Y collision
|
||||
//hit.prevPos.y = hit.pos.y;
|
||||
hit_pos.y += yi_step;
|
||||
if(tcross_y > max_distance)
|
||||
return false;
|
||||
tcross_y += tdelta_y;
|
||||
}
|
||||
else
|
||||
{
|
||||
// Z collision (duplicate code)
|
||||
//hit.prevPos.z = hit.pos.z;
|
||||
hit_pos.z += zi_step;
|
||||
if(tcross_z > max_distance)
|
||||
return false;
|
||||
tcross_z += tdelta_z;
|
||||
}
|
||||
}
|
||||
|
||||
} while(!predicate(hit_pos, predicate_context));
|
||||
|
||||
out_hit_pos = hit_pos;
|
||||
out_prev_pos = hit_prev_pos;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
18
voxel_raycast.h
Normal file
18
voxel_raycast.h
Normal file
@ -0,0 +1,18 @@
|
||||
#include <vector3.h>
|
||||
#include "vector3i.h"
|
||||
|
||||
// TODO Having a C++11 lambda would be nice...
|
||||
// pos: voxel position
|
||||
// context: arguments to carry (as a lamdbda capture)
|
||||
typedef bool(*VoxelPredicate)(Vector3i pos, void * context);
|
||||
|
||||
bool voxel_raycast(
|
||||
Vector3 ray_origin,
|
||||
Vector3 ray_direction,
|
||||
VoxelPredicate predicate,
|
||||
void * predicate_context, // Handle that one with care
|
||||
real_t max_distance,
|
||||
Vector3i & out_hit_pos,
|
||||
Vector3i & out_prev_pos
|
||||
);
|
||||
|
@ -1,6 +1,7 @@
|
||||
#include "voxel_terrain.h"
|
||||
#include <scene/3d/mesh_instance.h>
|
||||
#include <os/os.h>
|
||||
#include "voxel_raycast.h"
|
||||
|
||||
VoxelTerrain::VoxelTerrain(): Node(), _min_y(-4), _max_y(4), _generate_collisions(true) {
|
||||
|
||||
@ -24,6 +25,10 @@ Ref<VoxelProvider> VoxelTerrain::get_provider() {
|
||||
return _provider;
|
||||
}
|
||||
|
||||
Ref<VoxelLibrary> VoxelTerrain::get_voxel_library() {
|
||||
return _mesher->get_library();
|
||||
}
|
||||
|
||||
void VoxelTerrain::set_generate_collisions(bool enabled) {
|
||||
_generate_collisions = enabled;
|
||||
}
|
||||
@ -212,6 +217,46 @@ void VoxelTerrain::update_block_mesh(Vector3i block_pos) {
|
||||
// }
|
||||
//}
|
||||
|
||||
static bool _raycast_binding_predicate(Vector3i pos, void *context) {
|
||||
|
||||
ERR_FAIL_COND_V(context == NULL, false);
|
||||
VoxelTerrain * terrain = (VoxelTerrain*)context;
|
||||
|
||||
Ref<VoxelLibrary> lib_ref = terrain->get_voxel_library();
|
||||
if(lib_ref.is_null())
|
||||
return false;
|
||||
const VoxelLibrary & lib = **lib_ref;
|
||||
|
||||
Ref<VoxelMap> map = terrain->get_map();
|
||||
// TODO In the future we may want to query more channels
|
||||
int v = map->get_voxel(pos, 0);
|
||||
if(lib.has_voxel(v) == false)
|
||||
return false;
|
||||
|
||||
const Voxel & voxel = lib.get_voxel_const(v);
|
||||
return !voxel.is_transparent();
|
||||
}
|
||||
|
||||
Variant VoxelTerrain::_raycast_binding(Vector3 origin, Vector3 direction, real_t max_distance) {
|
||||
|
||||
// TODO Transform input if the terrain is rotated (in the future it can be made a Spatial node)
|
||||
|
||||
Vector3i hit_pos;
|
||||
Vector3i prev_pos;
|
||||
|
||||
if(voxel_raycast(origin, direction, _raycast_binding_predicate, this, max_distance, hit_pos, prev_pos)) {
|
||||
|
||||
Dictionary hit = Dictionary();
|
||||
hit["position"] = hit_pos.to_vec3();
|
||||
hit["prev_position"] = prev_pos.to_vec3();
|
||||
return hit;
|
||||
}
|
||||
else {
|
||||
return Variant(); // Null dictionary, no alloc
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void VoxelTerrain::_bind_methods() {
|
||||
|
||||
ClassDB::bind_method(D_METHOD("set_provider", "provider:VoxelProvider"), &VoxelTerrain::set_provider);
|
||||
@ -231,5 +276,7 @@ void VoxelTerrain::_bind_methods() {
|
||||
|
||||
ClassDB::bind_method(D_METHOD("force_load_blocks", "center", "extents"), &VoxelTerrain::_force_load_blocks_binding);
|
||||
|
||||
ClassDB::bind_method(D_METHOD("raycast:Dictionary", "origin", "direction", "max_distance"), &VoxelTerrain::_raycast_binding, DEFVAL(100));
|
||||
|
||||
}
|
||||
|
||||
|
@ -27,6 +27,7 @@ public:
|
||||
|
||||
Ref<VoxelMesher> get_mesher() { return _mesher; }
|
||||
Ref<VoxelMap> get_map() { return _map; }
|
||||
Ref<VoxelLibrary> get_voxel_library();
|
||||
|
||||
protected:
|
||||
void _notification(int p_what);
|
||||
@ -44,6 +45,7 @@ protected:
|
||||
Vector3 _voxel_to_block_binding(Vector3 pos) { return Vector3i(VoxelMap::voxel_to_block(pos)).to_vec3(); }
|
||||
Vector3 _block_to_voxel_binding(Vector3 pos) { return Vector3i(VoxelMap::block_to_voxel(pos)).to_vec3(); }
|
||||
void _force_load_blocks_binding(Vector3 center, Vector3 extents) { force_load_blocks(center, extents); }
|
||||
Variant _raycast_binding(Vector3 origin, Vector3 direction, real_t max_distance);
|
||||
|
||||
private:
|
||||
// Parameters
|
||||
|
Loading…
Reference in New Issue
Block a user