2019-05-03 16:02:10 -07:00
|
|
|
#include "voxel_lod_terrain.h"
|
|
|
|
#include "../math/rect3i.h"
|
2019-08-23 17:44:27 -07:00
|
|
|
#include "../streams/voxel_stream_file.h"
|
2019-05-04 17:09:12 -07:00
|
|
|
#include "../util/profiling_clock.h"
|
2019-05-03 16:02:10 -07:00
|
|
|
#include "voxel_map.h"
|
|
|
|
#include "voxel_mesh_updater.h"
|
2019-08-23 17:44:27 -07:00
|
|
|
#include <core/core_string_names.h>
|
2019-05-03 16:02:10 -07:00
|
|
|
#include <core/engine.h>
|
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
#ifdef TOOLS_ENABLED
|
|
|
|
#include <scene/3d/mesh_instance.h>
|
|
|
|
#endif
|
|
|
|
|
2019-05-19 10:27:49 -07:00
|
|
|
const uint32_t MAIN_THREAD_MESHING_BUDGET_MS = 8;
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
VoxelLodTerrain::VoxelLodTerrain() {
|
2019-08-23 17:44:27 -07:00
|
|
|
// Note: don't do anything heavy in the constructor.
|
|
|
|
// Godot may create and destroy dozens of instances of all node types on startup,
|
|
|
|
// due to how ClassDB gets its default values.
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
print_line("Construct VoxelLodTerrain");
|
|
|
|
|
|
|
|
_lods[0].map.instance();
|
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
set_lod_count(4);
|
2019-05-04 19:56:18 -07:00
|
|
|
set_lod_split_scale(3);
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
VoxelLodTerrain::~VoxelLodTerrain() {
|
|
|
|
|
|
|
|
print_line("Destroy VoxelLodTerrain");
|
|
|
|
|
2019-05-25 08:07:38 -07:00
|
|
|
if (_stream_thread) {
|
|
|
|
memdelete(_stream_thread);
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_block_updater) {
|
|
|
|
memdelete(_block_updater);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ref<Material> VoxelLodTerrain::get_material() const {
|
|
|
|
return _material;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::set_material(Ref<Material> p_material) {
|
|
|
|
_material = p_material;
|
|
|
|
}
|
|
|
|
|
2019-05-25 08:07:38 -07:00
|
|
|
Ref<VoxelStream> VoxelLodTerrain::get_stream() const {
|
|
|
|
return _stream;
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
2019-06-17 22:24:56 -07:00
|
|
|
unsigned int VoxelLodTerrain::get_block_size() const {
|
2019-05-03 16:02:10 -07:00
|
|
|
return _lods[0].map->get_block_size();
|
|
|
|
}
|
|
|
|
|
2019-06-17 22:24:56 -07:00
|
|
|
unsigned int VoxelLodTerrain::get_block_size_pow2() const {
|
2019-05-03 16:02:10 -07:00
|
|
|
return _lods[0].map->get_block_size_pow2();
|
|
|
|
}
|
|
|
|
|
2019-05-25 08:07:38 -07:00
|
|
|
void VoxelLodTerrain::set_stream(Ref<VoxelStream> p_stream) {
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
if (p_stream == _stream) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_stream.is_valid()) {
|
|
|
|
if (_stream->is_connected(CoreStringNames::get_singleton()->changed, this, "_on_stream_params_changed")) {
|
|
|
|
_stream->disconnect(CoreStringNames::get_singleton()->changed, this, "_on_stream_params_changed");
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
2019-08-23 17:44:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
_stream = p_stream;
|
|
|
|
|
|
|
|
if (_stream.is_valid()) {
|
|
|
|
_stream->connect(CoreStringNames::get_singleton()->changed, this, "_on_stream_params_changed");
|
|
|
|
}
|
|
|
|
|
|
|
|
_on_stream_params_changed();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::_on_stream_params_changed() {
|
|
|
|
|
|
|
|
stop_streamer();
|
|
|
|
|
|
|
|
bool was_updater_running = _block_updater != nullptr;
|
|
|
|
stop_updater();
|
|
|
|
|
|
|
|
Ref<VoxelStreamFile> file_stream = _stream;
|
|
|
|
if (file_stream.is_valid()) {
|
|
|
|
|
|
|
|
int stream_block_size_po2 = file_stream->get_block_size_po2();
|
|
|
|
_set_block_size_po2(stream_block_size_po2);
|
|
|
|
|
|
|
|
int stream_lod_count = file_stream->get_lod_count();
|
|
|
|
_set_lod_count(min(stream_lod_count, get_lod_count()));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_stream.is_valid()) {
|
|
|
|
start_streamer();
|
|
|
|
}
|
|
|
|
if (was_updater_running) {
|
|
|
|
start_updater();
|
|
|
|
}
|
|
|
|
|
|
|
|
// The whole map might change, so make all area dirty
|
|
|
|
// TODO Actually, we should regenerate the whole map, not just update all its blocks
|
|
|
|
make_all_view_dirty_deferred();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::set_block_size_po2(unsigned int p_block_size_po2) {
|
|
|
|
|
|
|
|
ERR_FAIL_COND(p_block_size_po2 < 1);
|
|
|
|
ERR_FAIL_COND(p_block_size_po2 > 32);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
unsigned int block_size_po2 = p_block_size_po2;
|
|
|
|
Ref<VoxelStreamFile> file_stream = _stream;
|
|
|
|
if (file_stream.is_valid()) {
|
|
|
|
block_size_po2 = file_stream->get_block_size_po2();
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
if (block_size_po2 == get_block_size_pow2()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool updater_was_running = _block_updater != nullptr;
|
|
|
|
|
|
|
|
stop_streamer();
|
|
|
|
stop_updater();
|
|
|
|
|
|
|
|
_set_block_size_po2(p_block_size_po2);
|
2019-08-29 14:55:02 -07:00
|
|
|
reset_maps();
|
2019-08-23 17:44:27 -07:00
|
|
|
|
|
|
|
if (_stream.is_valid()) {
|
|
|
|
start_streamer();
|
|
|
|
}
|
|
|
|
if (updater_was_running) {
|
|
|
|
start_updater();
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
void VoxelLodTerrain::_set_block_size_po2(int p_block_size_po2) {
|
|
|
|
_lods[0].map->create(p_block_size_po2, 0);
|
|
|
|
}
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
void VoxelLodTerrain::make_all_view_dirty_deferred() {
|
2019-06-17 22:24:56 -07:00
|
|
|
for (unsigned int i = 0; i < get_lod_count(); ++i) {
|
2019-05-03 16:02:10 -07:00
|
|
|
Lod &lod = _lods[i];
|
|
|
|
lod.last_view_distance_blocks = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int VoxelLodTerrain::get_view_distance() const {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
void VoxelLodTerrain::set_view_distance(unsigned int p_distance_in_voxels) {
|
|
|
|
|
|
|
|
ERR_FAIL_COND(p_distance_in_voxels == 0);
|
|
|
|
ERR_FAIL_COND(p_distance_in_voxels > 8192);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
// Note: this is a hint distance, the terrain will attempt to have this radius filled with loaded voxels.
|
|
|
|
// It is possible for blocks to still load beyond that distance.
|
|
|
|
_view_distance_voxels = p_distance_in_voxels;
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
Spatial *VoxelLodTerrain::get_viewer() const {
|
|
|
|
if (_viewer_path.is_empty()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
Node *node = get_node(_viewer_path);
|
|
|
|
if (node == nullptr) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return Object::cast_to<Spatial>(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::immerge_block(Vector3i block_pos, unsigned int lod_index) {
|
|
|
|
|
|
|
|
ERR_FAIL_COND(lod_index >= get_lod_count());
|
|
|
|
ERR_FAIL_COND(_lods[lod_index].map.is_null());
|
|
|
|
|
|
|
|
Lod &lod = _lods[lod_index];
|
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
// TODO Schedule block saving if modified, it's supported now!
|
2019-05-03 16:02:10 -07:00
|
|
|
lod.map->remove_block(block_pos, VoxelMap::NoAction());
|
|
|
|
|
2019-05-09 11:11:17 -07:00
|
|
|
lod.loading_blocks.erase(block_pos);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
// Blocks in the update queue will be cancelled in _process,
|
|
|
|
// because it's too expensive to linear-search all blocks for each block
|
|
|
|
|
|
|
|
// No need to remove things from blocks_pending_load,
|
|
|
|
// This vector is filled and cleared immediately in the main process.
|
|
|
|
// It is a member only to re-use its capacity memory over frames.
|
|
|
|
}
|
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
void VoxelLodTerrain::start_updater() {
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
ERR_FAIL_COND(_block_updater != nullptr);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
// TODO Thread-safe way to change those parameters
|
|
|
|
VoxelMeshUpdater::MeshingParams params;
|
|
|
|
params.smooth_surface = true;
|
|
|
|
|
2019-05-19 10:27:49 -07:00
|
|
|
_block_updater = memnew(VoxelMeshUpdater(2, params));
|
2019-08-23 17:44:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::stop_updater() {
|
|
|
|
|
|
|
|
struct ResetMeshStateAction {
|
|
|
|
void operator()(VoxelBlock *block) {
|
|
|
|
if (block->get_mesh_state() == VoxelBlock::MESH_UPDATE_SENT) {
|
|
|
|
block->set_mesh_state(VoxelBlock::MESH_UPDATE_NOT_SENT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if (_block_updater) {
|
|
|
|
memdelete(_block_updater);
|
|
|
|
_block_updater = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
_blocks_pending_main_thread_update.clear();
|
|
|
|
|
|
|
|
for (int i = 0; i < MAX_LOD; ++i) {
|
|
|
|
|
|
|
|
Lod &lod = _lods[i];
|
|
|
|
lod.blocks_pending_update.clear();
|
|
|
|
|
2019-08-24 14:38:27 -07:00
|
|
|
if (lod.map.is_valid()) {
|
|
|
|
ResetMeshStateAction a;
|
|
|
|
lod.map->for_all_blocks(a);
|
|
|
|
}
|
2019-08-23 17:44:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::start_streamer() {
|
|
|
|
|
|
|
|
ERR_FAIL_COND(_stream_thread != nullptr);
|
|
|
|
ERR_FAIL_COND(_stream.is_null());
|
|
|
|
|
|
|
|
_stream_thread = memnew(VoxelDataLoader(1, _stream, get_block_size_pow2()));
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
void VoxelLodTerrain::stop_streamer() {
|
|
|
|
|
|
|
|
if (_stream_thread) {
|
|
|
|
memdelete(_stream_thread);
|
|
|
|
_stream_thread = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < MAX_LOD; ++i) {
|
|
|
|
Lod &lod = _lods[i];
|
|
|
|
lod.blocks_to_load.clear();
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::set_lod_split_scale(float p_lod_split_scale) {
|
2019-08-29 14:55:02 -07:00
|
|
|
|
|
|
|
if (p_lod_split_scale == _lod_split_scale) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_lod_split_scale = p_lod_split_scale;
|
|
|
|
|
|
|
|
for (Map<Vector3i, OctreeItem>::Element *E = _lod_octrees.front(); E; E = E->next()) {
|
|
|
|
|
|
|
|
OctreeItem &item = E->value();
|
|
|
|
item.octree.set_split_scale(_lod_split_scale);
|
|
|
|
|
|
|
|
// Because `set_split_scale` may clamp it...
|
|
|
|
_lod_split_scale = item.octree.get_split_scale();
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
float VoxelLodTerrain::get_lod_split_scale() const {
|
2019-08-29 14:55:02 -07:00
|
|
|
return _lod_split_scale;
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
void VoxelLodTerrain::set_lod_count(int p_lod_count) {
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
ERR_FAIL_COND(p_lod_count >= MAX_LOD);
|
2019-08-29 14:55:02 -07:00
|
|
|
ERR_FAIL_COND(p_lod_count < 1);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
if (get_lod_count() != p_lod_count) {
|
2019-08-23 17:44:27 -07:00
|
|
|
_set_lod_count(p_lod_count);
|
|
|
|
}
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
void VoxelLodTerrain::_set_lod_count(int p_lod_count) {
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
CRASH_COND(p_lod_count >= MAX_LOD);
|
|
|
|
CRASH_COND(p_lod_count < 1);
|
|
|
|
|
|
|
|
_lod_count = p_lod_count;
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
LodOctree<bool>::NoDestroyAction nda;
|
2019-08-29 14:55:02 -07:00
|
|
|
|
|
|
|
for (Map<Vector3i, OctreeItem>::Element *E = _lod_octrees.front(); E; E = E->next()) {
|
|
|
|
OctreeItem &item = E->value();
|
|
|
|
item.octree.create_from_lod_count(get_block_size(), p_lod_count, nda);
|
|
|
|
#ifdef TOOLS_ENABLED
|
|
|
|
destroy_octree_debug_box(item, E->key());
|
|
|
|
create_octree_debug_box(item, E->key());
|
|
|
|
#endif
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
reset_maps();
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
void VoxelLodTerrain::reset_maps() {
|
2019-08-29 14:55:02 -07:00
|
|
|
// Clears all blocks and reconfigures maps to account for new LOD count and block sizes
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
for (int lod_index = 0; lod_index < MAX_LOD; ++lod_index) {
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
Lod &lod = _lods[lod_index];
|
|
|
|
|
|
|
|
// Instance new maps if we have more lods, or clear them otherwise
|
|
|
|
if (lod_index < get_lod_count()) {
|
|
|
|
|
|
|
|
if (lod.map.is_null()) {
|
|
|
|
lod.map.instance();
|
|
|
|
}
|
|
|
|
lod.map->create(get_block_size_pow2(), lod_index);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
if (lod.map.is_valid()) {
|
|
|
|
lod.map.unref();
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
int VoxelLodTerrain::get_lod_count() const {
|
2019-08-29 14:55:02 -07:00
|
|
|
return _lod_count;
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
2019-08-25 05:04:49 -07:00
|
|
|
void VoxelLodTerrain::set_generate_collisions(bool enabled) {
|
|
|
|
_generate_collisions = enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::set_collision_lod_count(int lod_count) {
|
|
|
|
_collision_lod_count = CLAMP(lod_count, -1, get_lod_count());
|
|
|
|
}
|
|
|
|
|
|
|
|
int VoxelLodTerrain::get_collision_lod_count() const {
|
|
|
|
return _collision_lod_count;
|
|
|
|
}
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
void VoxelLodTerrain::set_viewer_path(NodePath path) {
|
|
|
|
_viewer_path = path;
|
|
|
|
}
|
|
|
|
|
|
|
|
NodePath VoxelLodTerrain::get_viewer_path() const {
|
|
|
|
return _viewer_path;
|
|
|
|
}
|
|
|
|
|
2019-05-07 16:57:34 -07:00
|
|
|
int VoxelLodTerrain::get_block_region_extent() const {
|
|
|
|
// This is the radius of blocks around the viewer in which we may load them.
|
|
|
|
// It depends on the LOD split scale, which tells how close to a block we need to be for it to subdivide.
|
|
|
|
// Each LOD is fractal so that value is the same for each of them.
|
2019-08-29 14:55:02 -07:00
|
|
|
return static_cast<int>(_lod_split_scale) * 2 + 2;
|
2019-05-07 16:57:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
Dictionary VoxelLodTerrain::get_block_info(Vector3 fbpos, unsigned int lod_index) const {
|
|
|
|
// Gets some info useful for debugging
|
|
|
|
Dictionary d;
|
|
|
|
ERR_FAIL_COND_V(lod_index >= get_lod_count(), d);
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
const Lod &lod = _lods[lod_index];
|
2019-05-07 16:57:34 -07:00
|
|
|
Vector3i bpos(fbpos);
|
|
|
|
|
|
|
|
bool meshed = false;
|
|
|
|
bool visible = false;
|
2019-05-09 11:11:17 -07:00
|
|
|
int loading_state = 0;
|
2019-05-07 16:57:34 -07:00
|
|
|
const VoxelBlock *block = lod.map->get_block(bpos);
|
2019-05-03 16:02:10 -07:00
|
|
|
if (block) {
|
2019-05-09 14:00:54 -07:00
|
|
|
meshed = block->has_been_meshed();
|
2019-05-07 16:57:34 -07:00
|
|
|
visible = block->is_visible();
|
2019-05-09 11:11:17 -07:00
|
|
|
loading_state = 2;
|
|
|
|
} else if (lod.loading_blocks.has(bpos)) {
|
|
|
|
loading_state = 1;
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
2019-05-07 16:57:34 -07:00
|
|
|
|
2019-05-20 12:49:46 -07:00
|
|
|
#ifdef TOOLS_ENABLED
|
|
|
|
int debug_unexpected_drop_time = -10000;
|
|
|
|
const int *ptr = lod.debug_unexpected_load_drop_time.getptr(bpos);
|
|
|
|
if (ptr) {
|
|
|
|
debug_unexpected_drop_time = *ptr;
|
|
|
|
}
|
|
|
|
d["debug_unexpected_drop_time"] = debug_unexpected_drop_time;
|
|
|
|
#endif
|
|
|
|
|
2019-05-09 11:11:17 -07:00
|
|
|
d["loading"] = loading_state;
|
2019-05-07 16:57:34 -07:00
|
|
|
d["meshed"] = meshed;
|
|
|
|
d["visible"] = visible;
|
|
|
|
return d;
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
2019-05-07 16:57:34 -07:00
|
|
|
Vector3 VoxelLodTerrain::voxel_to_block_position(Vector3 vpos, unsigned int lod_index) const {
|
|
|
|
ERR_FAIL_COND_V(lod_index >= get_lod_count(), Vector3());
|
2019-05-03 16:02:10 -07:00
|
|
|
const Lod &lod = _lods[lod_index];
|
2019-05-07 16:57:34 -07:00
|
|
|
Vector3i bpos = lod.map->voxel_to_block(Vector3i(vpos)) >> lod_index;
|
|
|
|
return bpos.to_vec3();
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::_notification(int p_what) {
|
|
|
|
|
2019-08-24 17:11:38 -07:00
|
|
|
struct SetWorldAction {
|
2019-05-03 16:02:10 -07:00
|
|
|
World *world;
|
2019-08-24 17:11:38 -07:00
|
|
|
SetWorldAction(World *w) :
|
2019-05-03 16:02:10 -07:00
|
|
|
world(w) {}
|
|
|
|
void operator()(VoxelBlock *block) {
|
2019-08-24 17:11:38 -07:00
|
|
|
block->set_world(world);
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-08-31 13:46:19 -07:00
|
|
|
struct SetParentVisibilityAction {
|
2019-05-03 16:02:10 -07:00
|
|
|
bool visible;
|
2019-08-31 13:46:19 -07:00
|
|
|
SetParentVisibilityAction(bool v) :
|
2019-05-03 16:02:10 -07:00
|
|
|
visible(v) {}
|
|
|
|
void operator()(VoxelBlock *block) {
|
2019-08-31 13:46:19 -07:00
|
|
|
block->set_parent_visible(visible);
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (p_what) {
|
|
|
|
|
|
|
|
case NOTIFICATION_ENTER_TREE:
|
2019-08-23 17:44:27 -07:00
|
|
|
if (_block_updater == nullptr) {
|
|
|
|
start_updater();
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
set_process(true);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NOTIFICATION_PROCESS:
|
|
|
|
if (!Engine::get_singleton()->is_editor_hint()) {
|
|
|
|
_process();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NOTIFICATION_EXIT_TREE:
|
|
|
|
break;
|
|
|
|
|
2019-08-14 17:21:45 -07:00
|
|
|
case NOTIFICATION_ENTER_WORLD: {
|
2019-08-24 17:11:38 -07:00
|
|
|
SetWorldAction ewa(*get_world());
|
2019-08-14 17:21:45 -07:00
|
|
|
for_all_blocks(ewa);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case NOTIFICATION_EXIT_WORLD: {
|
2019-08-24 17:11:38 -07:00
|
|
|
SetWorldAction ewa(nullptr);
|
2019-08-14 17:21:45 -07:00
|
|
|
for_all_blocks(ewa);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case NOTIFICATION_VISIBILITY_CHANGED: {
|
2019-08-31 13:46:19 -07:00
|
|
|
SetParentVisibilityAction sva(is_visible());
|
2019-08-14 17:21:45 -07:00
|
|
|
for_all_blocks(sva);
|
|
|
|
} break;
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
// TODO Listen for transform changes
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-19 13:04:26 -07:00
|
|
|
Vector3 VoxelLodTerrain::get_viewer_pos(Vector3 &out_direction) const {
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
if (Engine::get_singleton()->is_editor_hint()) {
|
|
|
|
|
|
|
|
// TODO Use editor's camera here
|
|
|
|
return Vector3();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
Spatial *viewer = get_viewer();
|
|
|
|
|
|
|
|
if (viewer) {
|
2019-05-19 13:04:26 -07:00
|
|
|
out_direction = -viewer->get_global_transform().basis.get_axis(Vector3::AXIS_Z);
|
2019-05-03 16:02:10 -07:00
|
|
|
return viewer->get_global_transform().origin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Vector3();
|
|
|
|
}
|
|
|
|
|
2019-05-10 10:56:48 -07:00
|
|
|
void VoxelLodTerrain::try_schedule_loading_with_neighbors(const Vector3i &p_bpos, unsigned int lod_index) {
|
2019-05-08 12:53:51 -07:00
|
|
|
CRASH_COND(lod_index >= get_lod_count());
|
|
|
|
Lod &lod = _lods[lod_index];
|
|
|
|
|
|
|
|
Vector3i bpos;
|
|
|
|
|
|
|
|
for (int y = -1; y < 2; ++y) {
|
|
|
|
for (int z = -1; z < 2; ++z) {
|
|
|
|
for (int x = -1; x < 2; ++x) {
|
|
|
|
|
|
|
|
bpos.x = p_bpos.x + x;
|
|
|
|
bpos.y = p_bpos.y + y;
|
|
|
|
bpos.z = p_bpos.z + z;
|
|
|
|
|
|
|
|
VoxelBlock *block = lod.map->get_block(bpos);
|
|
|
|
|
|
|
|
if (block == nullptr) {
|
2019-05-09 11:11:17 -07:00
|
|
|
if (!lod.loading_blocks.has(bpos)) {
|
2019-05-08 12:53:51 -07:00
|
|
|
lod.blocks_to_load.push_back(bpos);
|
2019-05-09 11:11:17 -07:00
|
|
|
lod.loading_blocks.insert(bpos);
|
2019-05-08 12:53:51 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-10 10:56:48 -07:00
|
|
|
bool VoxelLodTerrain::check_block_loaded_and_updated(const Vector3i &p_bpos, unsigned int lod_index) {
|
|
|
|
CRASH_COND(lod_index >= get_lod_count());
|
|
|
|
Lod &lod = _lods[lod_index];
|
|
|
|
|
|
|
|
VoxelBlock *block = lod.map->get_block(p_bpos);
|
|
|
|
|
|
|
|
if (block == nullptr) {
|
|
|
|
try_schedule_loading_with_neighbors(p_bpos, lod_index);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!block->has_been_meshed()) {
|
|
|
|
if (!block->is_mesh_update_scheduled()) {
|
2019-05-25 07:50:01 -07:00
|
|
|
if (lod.map->is_block_surrounded(block->position)) {
|
2019-05-10 10:56:48 -07:00
|
|
|
|
2019-05-25 07:50:01 -07:00
|
|
|
lod.blocks_pending_update.push_back(block->position);
|
2019-05-10 10:56:48 -07:00
|
|
|
block->set_mesh_state(VoxelBlock::MESH_UPDATE_NOT_SENT);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
try_schedule_loading_with_neighbors(p_bpos, lod_index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-05-20 12:34:41 -07:00
|
|
|
//static void remove_positions_outside_box(
|
|
|
|
// std::vector<Vector3i> &positions,
|
|
|
|
// Rect3i box,
|
|
|
|
// Set<Vector3i> &position_set) {
|
|
|
|
|
2019-06-17 22:24:56 -07:00
|
|
|
// for (unsigned int i = 0; i < positions.size(); ++i) {
|
2019-05-20 12:34:41 -07:00
|
|
|
// const Vector3i bpos = positions[i];
|
|
|
|
// if (!box.contains(bpos)) {
|
|
|
|
// int last = positions.size() - 1;
|
|
|
|
// positions[i] = positions[last];
|
|
|
|
// positions.resize(last);
|
|
|
|
// position_set.erase(bpos);
|
|
|
|
// --i;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
//}
|
2019-05-09 11:11:17 -07:00
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
void VoxelLodTerrain::_process() {
|
|
|
|
|
|
|
|
if (get_lod_count() == 0) {
|
|
|
|
// If there isn't a LOD 0, there is nothing to load
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
OS &os = *OS::get_singleton();
|
|
|
|
|
2019-05-19 13:04:26 -07:00
|
|
|
Vector3 viewer_direction;
|
|
|
|
Vector3 viewer_pos = get_viewer_pos(viewer_direction);
|
2019-05-03 16:02:10 -07:00
|
|
|
Vector3i viewer_block_pos = _lods[0].map->voxel_to_block(viewer_pos);
|
|
|
|
|
2019-05-04 17:09:12 -07:00
|
|
|
ProfilingClock profiling_clock;
|
|
|
|
|
2019-05-12 08:30:45 -07:00
|
|
|
_stats.dropped_block_loads = 0;
|
|
|
|
_stats.dropped_block_meshs = 0;
|
2019-08-29 14:55:02 -07:00
|
|
|
_stats.blocked_lods = 0;
|
2019-05-12 08:30:45 -07:00
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
// Here we go...
|
|
|
|
|
2019-05-08 12:53:51 -07:00
|
|
|
// Remove blocks falling out of block region extent
|
2019-05-03 16:02:10 -07:00
|
|
|
{
|
2019-08-29 14:55:02 -07:00
|
|
|
// TODO Could it actually be enough to have a rolling update on all blocks?
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
// This should be the same distance relatively to each LOD
|
2019-05-08 12:53:51 -07:00
|
|
|
int block_region_extent = get_block_region_extent();
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
for (unsigned int lod_index = 0; lod_index < get_lod_count(); ++lod_index) {
|
|
|
|
Lod &lod = _lods[lod_index];
|
|
|
|
|
|
|
|
// Each LOD keeps a box of loaded blocks, and only some of the blocks will get polygonized.
|
|
|
|
// The player can edit them so changes can be propagated to lower lods.
|
|
|
|
|
2019-06-17 22:24:56 -07:00
|
|
|
unsigned int block_size_po2 = _lods[0].map->get_block_size_pow2() + lod_index;
|
2019-05-03 16:02:10 -07:00
|
|
|
Vector3i viewer_block_pos_within_lod = VoxelMap::voxel_to_block_b(viewer_pos, block_size_po2);
|
|
|
|
|
2019-05-08 12:53:51 -07:00
|
|
|
Rect3i new_box = Rect3i::from_center_extents(viewer_block_pos_within_lod, Vector3i(block_region_extent));
|
|
|
|
Rect3i prev_box = Rect3i::from_center_extents(lod.last_viewer_block_pos, Vector3i(lod.last_view_distance_blocks));
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
// Eliminate pending blocks that aren't needed
|
2019-05-20 12:34:41 -07:00
|
|
|
|
|
|
|
// No need to do it on those arrays, they are always clear at this point.
|
|
|
|
// Let's assert so it will pop on your face the day that assumption changes
|
|
|
|
CRASH_COND(!lod.blocks_to_load.empty());
|
|
|
|
CRASH_COND(!lod.blocks_pending_update.empty());
|
|
|
|
//remove_positions_outside_box(lod.blocks_to_load, new_box, lod.loading_blocks);
|
|
|
|
//remove_positions_outside_box(lod.blocks_pending_update, new_box, lod.loading_blocks);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
if (prev_box != new_box) {
|
|
|
|
|
|
|
|
Rect3i bounds = Rect3i::get_bounding_box(prev_box, new_box);
|
|
|
|
Vector3i max = bounds.pos + bounds.size;
|
|
|
|
|
|
|
|
// TODO This will explode if the player teleports!
|
|
|
|
// There is a smarter way to only iterate relevant blocks
|
|
|
|
|
|
|
|
Vector3i pos;
|
|
|
|
for (pos.z = bounds.pos.z; pos.z < max.z; ++pos.z) {
|
|
|
|
for (pos.y = bounds.pos.y; pos.y < max.y; ++pos.y) {
|
|
|
|
for (pos.x = bounds.pos.x; pos.x < max.x; ++pos.x) {
|
|
|
|
|
|
|
|
bool prev_contains = prev_box.contains(pos);
|
|
|
|
bool new_contains = new_box.contains(pos);
|
|
|
|
|
|
|
|
if (prev_contains && !new_contains) {
|
|
|
|
// Unload block
|
|
|
|
immerge_block(pos, lod_index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lod.last_viewer_block_pos = viewer_block_pos_within_lod;
|
2019-05-08 12:53:51 -07:00
|
|
|
lod.last_view_distance_blocks = block_region_extent;
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
// Create and remove octrees in a grid around the viewer
|
2019-05-08 12:53:51 -07:00
|
|
|
{
|
2019-08-29 14:55:02 -07:00
|
|
|
// TODO Investigate if multi-octree can produce cracks in the terrain
|
|
|
|
// TODO Need to work when lod count changes at runtime
|
|
|
|
|
|
|
|
unsigned int octree_size_po2 = get_block_size_pow2() + get_lod_count() - 1;
|
|
|
|
unsigned int octree_region_extent = 1 + _view_distance_voxels / (1 << octree_size_po2);
|
|
|
|
|
|
|
|
Vector3i viewer_octree_pos = VoxelMap::voxel_to_block_b(viewer_pos, octree_size_po2);
|
|
|
|
|
|
|
|
Rect3i new_box = Rect3i::from_center_extents(viewer_octree_pos, Vector3i(octree_region_extent));
|
|
|
|
Rect3i prev_box = Rect3i::from_center_extents(_last_viewer_octree_position, Vector3i(_last_octree_region_extent));
|
|
|
|
|
|
|
|
if (new_box != prev_box) {
|
|
|
|
|
|
|
|
struct CleanOctreeAction {
|
|
|
|
VoxelLodTerrain *self;
|
|
|
|
Vector3i block_offset_lod0;
|
|
|
|
|
|
|
|
void operator()(LodOctree<bool>::Node *node, unsigned int lod_index) {
|
|
|
|
Lod &lod = self->_lods[lod_index];
|
|
|
|
|
|
|
|
Vector3i bpos = node->position;
|
|
|
|
bpos += block_offset_lod0 >> lod_index;
|
|
|
|
|
|
|
|
VoxelBlock *block = lod.map->get_block(bpos);
|
|
|
|
if (block) {
|
|
|
|
block->set_visible(false);
|
|
|
|
}
|
2019-05-08 12:53:51 -07:00
|
|
|
}
|
2019-08-29 14:55:02 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ExitAction {
|
|
|
|
VoxelLodTerrain *self;
|
|
|
|
void operator()(const Vector3i &pos) {
|
|
|
|
|
|
|
|
Map<Vector3i, OctreeItem>::Element *E = self->_lod_octrees.find(pos);
|
|
|
|
if (E == nullptr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
OctreeItem &item = E->value();
|
|
|
|
Vector3i block_pos_maxlod = E->key();
|
|
|
|
|
|
|
|
#ifdef TOOLS_ENABLED
|
|
|
|
self->destroy_octree_debug_box(item, block_pos_maxlod);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int last_lod_index = self->get_lod_count() - 1;
|
|
|
|
|
|
|
|
// We just drop the octree and hide blocks it was considering as visible.
|
|
|
|
// Normally such octrees shouldn't bee too deep as they will likely be at the edge
|
|
|
|
// of the loaded area, unless the player teleported far away.
|
|
|
|
CleanOctreeAction a;
|
|
|
|
a.self = self;
|
|
|
|
a.block_offset_lod0 = block_pos_maxlod << last_lod_index;
|
|
|
|
item.octree.clear(a);
|
|
|
|
|
|
|
|
self->_lod_octrees.erase(E);
|
2019-05-08 12:53:51 -07:00
|
|
|
}
|
2019-08-29 14:55:02 -07:00
|
|
|
};
|
2019-05-08 12:53:51 -07:00
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
struct EnterAction {
|
|
|
|
VoxelLodTerrain *self;
|
|
|
|
int block_size;
|
|
|
|
void operator()(const Vector3i &pos) {
|
|
|
|
CRASH_COND(self->_lod_octrees.has(pos));
|
|
|
|
|
|
|
|
Map<Vector3i, OctreeItem>::Element *E = self->_lod_octrees.insert(pos, OctreeItem());
|
|
|
|
CRASH_COND(E == nullptr);
|
|
|
|
OctreeItem &item = E->value();
|
|
|
|
LodOctree<bool>::NoDestroyAction nda;
|
|
|
|
item.octree.create_from_lod_count(block_size, self->get_lod_count(), nda);
|
|
|
|
|
|
|
|
#ifdef TOOLS_ENABLED
|
|
|
|
self->create_octree_debug_box(item, pos);
|
|
|
|
#endif
|
2019-05-08 12:53:51 -07:00
|
|
|
}
|
2019-08-29 14:55:02 -07:00
|
|
|
};
|
2019-05-08 12:53:51 -07:00
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
ExitAction exit_action;
|
|
|
|
exit_action.self = this;
|
|
|
|
|
|
|
|
EnterAction enter_action;
|
|
|
|
enter_action.self = this;
|
|
|
|
enter_action.block_size = get_block_size();
|
|
|
|
|
|
|
|
Rect3i::check_enters_and_exits(prev_box, new_box, exit_action, enter_action);
|
|
|
|
}
|
|
|
|
|
|
|
|
_last_viewer_octree_position = viewer_octree_pos;
|
|
|
|
_last_octree_region_extent = octree_region_extent;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find which blocks we need to load and see, within each octree
|
|
|
|
{
|
|
|
|
// TODO Maintain a vector to make iteration faster?
|
|
|
|
for (Map<Vector3i, OctreeItem>::Element *E = _lod_octrees.front(); E; E = E->next()) {
|
|
|
|
|
|
|
|
OctreeItem &item = E->value();
|
|
|
|
Vector3i block_pos_maxlod = E->key();
|
|
|
|
Vector3i block_offset_lod0 = block_pos_maxlod << (get_lod_count() - 1);
|
|
|
|
|
|
|
|
struct SubdivideAction {
|
|
|
|
VoxelLodTerrain *self;
|
|
|
|
Vector3i block_offset_lod0;
|
|
|
|
unsigned int blocked_count = 0;
|
|
|
|
|
|
|
|
bool can_do_root(unsigned int lod_index) {
|
|
|
|
Vector3i offset = block_offset_lod0 >> lod_index;
|
|
|
|
return self->check_block_loaded_and_updated(offset, lod_index);
|
2019-05-08 12:53:51 -07:00
|
|
|
}
|
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
bool can_do_children(LodOctree<bool>::Node *node, unsigned int child_lod_index) {
|
|
|
|
|
|
|
|
Vector3i offset = block_offset_lod0 >> child_lod_index;
|
|
|
|
bool can = true;
|
|
|
|
|
|
|
|
// Can only subdivide if higher detail meshes are ready to be shown, otherwise it will produce holes
|
|
|
|
for (int i = 0; i < 8; ++i) {
|
|
|
|
|
|
|
|
// Get block pos local-to-region
|
|
|
|
Vector3i child_pos = LodOctree<bool>::get_child_position(node->position, i);
|
|
|
|
|
|
|
|
// Convert to local-to-terrain
|
|
|
|
child_pos += offset;
|
|
|
|
|
|
|
|
// We have to ping ALL children, because the reason we are here is we want them loaded
|
|
|
|
can &= self->check_block_loaded_and_updated(child_pos, child_lod_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!can) {
|
|
|
|
++blocked_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return can;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator()(LodOctree<bool>::Node *node, unsigned int lod_index) {
|
|
|
|
|
|
|
|
Lod &lod = self->_lods[lod_index];
|
|
|
|
|
|
|
|
Vector3i bpos = node->position;
|
|
|
|
bpos += block_offset_lod0 >> lod_index;
|
|
|
|
|
|
|
|
VoxelBlock *block = lod.map->get_block(bpos);
|
|
|
|
|
|
|
|
CRASH_COND(block == nullptr);
|
|
|
|
CRASH_COND(!block->has_been_meshed()); // Never show a block that hasn't been meshed
|
|
|
|
|
|
|
|
block->set_visible(true);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct UnsubdivideAction {
|
|
|
|
VoxelLodTerrain *self;
|
|
|
|
Vector3i block_offset_lod0;
|
|
|
|
unsigned int blocked_count = 0;
|
|
|
|
|
|
|
|
bool can_do(LodOctree<bool>::Node *node, unsigned int parent_lod_index) {
|
|
|
|
|
|
|
|
// Can only unsubdivide if the parent mesh is ready
|
|
|
|
Vector3i bpos = node->position;
|
|
|
|
bpos += block_offset_lod0 >> parent_lod_index;
|
|
|
|
|
|
|
|
bool can = self->check_block_loaded_and_updated(bpos, parent_lod_index);
|
2019-05-08 12:53:51 -07:00
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
if (!can) {
|
|
|
|
++blocked_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return can;
|
|
|
|
}
|
|
|
|
|
|
|
|
void operator()(LodOctree<bool>::Node *node, unsigned int lod_index) {
|
2019-05-08 12:53:51 -07:00
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
Lod &lod = self->_lods[lod_index];
|
2019-05-08 12:53:51 -07:00
|
|
|
|
2019-08-29 14:55:02 -07:00
|
|
|
Vector3i bpos = node->position;
|
|
|
|
bpos += block_offset_lod0 >> lod_index;
|
|
|
|
|
|
|
|
VoxelBlock *block = lod.map->get_block(bpos);
|
|
|
|
if (block) {
|
|
|
|
block->set_visible(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
SubdivideAction subdivide_action;
|
|
|
|
subdivide_action.self = this;
|
|
|
|
subdivide_action.block_offset_lod0 = block_offset_lod0;
|
|
|
|
|
|
|
|
UnsubdivideAction unsubdivide_action;
|
|
|
|
unsubdivide_action.self = this;
|
|
|
|
unsubdivide_action.block_offset_lod0 = block_offset_lod0;
|
|
|
|
|
|
|
|
Vector3 relative_viewer_pos = viewer_pos - get_block_size() * block_offset_lod0.to_vec3();
|
|
|
|
item.octree.update(relative_viewer_pos, subdivide_action, unsubdivide_action);
|
|
|
|
|
|
|
|
// Ideally, this stat should stabilize to zero.
|
|
|
|
// If not, something in block management prevents LODs to properly show up and should be fixed.
|
|
|
|
_stats.blocked_lods += subdivide_action.blocked_count + unsubdivide_action.blocked_count;
|
|
|
|
}
|
2019-05-08 12:53:51 -07:00
|
|
|
}
|
|
|
|
|
2019-08-25 10:47:43 -07:00
|
|
|
_stats.time_detect_required_blocks = profiling_clock.restart();
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
// Send block loading requests
|
|
|
|
{
|
2019-05-20 12:48:58 -07:00
|
|
|
VoxelDataLoader::Input input;
|
|
|
|
input.priority_position = viewer_block_pos;
|
|
|
|
input.priority_direction = viewer_direction;
|
2019-08-29 17:08:06 -07:00
|
|
|
input.use_exclusive_region = true;
|
|
|
|
// The last LOD may spread until end of view distance, it should not be discarded
|
|
|
|
input.exclusive_region_max_lod = get_lod_count() - 1;
|
2019-05-12 08:33:25 -07:00
|
|
|
input.exclusive_region_extent = get_block_region_extent();
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
for (unsigned int lod_index = 0; lod_index < get_lod_count(); ++lod_index) {
|
|
|
|
Lod &lod = _lods[lod_index];
|
|
|
|
|
2019-06-17 22:24:56 -07:00
|
|
|
for (unsigned int i = 0; i < lod.blocks_to_load.size(); ++i) {
|
2019-05-20 12:48:58 -07:00
|
|
|
VoxelDataLoader::InputBlock input_block;
|
|
|
|
input_block.position = lod.blocks_to_load[i];
|
2019-05-03 16:02:10 -07:00
|
|
|
input_block.lod = lod_index;
|
2019-05-20 12:48:58 -07:00
|
|
|
input.blocks.push_back(input_block);
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
lod.blocks_to_load.clear();
|
|
|
|
}
|
|
|
|
|
2019-05-08 12:53:51 -07:00
|
|
|
//print_line(String("Sending {0}").format(varray(input.blocks_to_emerge.size())));
|
2019-05-25 08:07:38 -07:00
|
|
|
_stream_thread->push(input);
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
2019-05-04 17:09:12 -07:00
|
|
|
_stats.time_request_blocks_to_load = profiling_clock.restart();
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
// Get block loading responses
|
|
|
|
// Note: if block loading is too fast, this can cause stutters.
|
|
|
|
// It should only happen on first load, though.
|
|
|
|
{
|
2019-05-20 12:48:58 -07:00
|
|
|
VoxelDataLoader::Output output;
|
2019-05-25 08:07:38 -07:00
|
|
|
_stream_thread->pop(output);
|
2019-05-27 17:10:50 -07:00
|
|
|
_stats.stream = output.stats;
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-05-08 12:53:51 -07:00
|
|
|
//print_line(String("Loaded {0} blocks").format(varray(output.emerged_blocks.size())));
|
|
|
|
|
2019-05-20 12:48:58 -07:00
|
|
|
for (int i = 0; i < output.blocks.size(); ++i) {
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-05-20 12:48:58 -07:00
|
|
|
const VoxelDataLoader::OutputBlock &ob = output.blocks[i];
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-05-20 12:48:58 -07:00
|
|
|
if (ob.lod >= get_lod_count()) {
|
2019-05-03 16:02:10 -07:00
|
|
|
// That block was requested at a time where LOD was higher... drop it
|
2019-05-12 08:30:45 -07:00
|
|
|
++_stats.dropped_block_loads;
|
2019-05-03 16:02:10 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-05-20 12:48:58 -07:00
|
|
|
Lod &lod = _lods[ob.lod];
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-25 07:14:10 -07:00
|
|
|
{
|
|
|
|
Set<Vector3i>::Element *E = lod.loading_blocks.find(ob.position);
|
|
|
|
if (E == nullptr) {
|
|
|
|
// That block was not requested, or is no longer needed. drop it...
|
|
|
|
++_stats.dropped_block_loads;
|
|
|
|
continue;
|
|
|
|
}
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-25 07:14:10 -07:00
|
|
|
lod.loading_blocks.erase(E);
|
|
|
|
}
|
2019-05-09 14:00:54 -07:00
|
|
|
|
2019-05-20 12:48:58 -07:00
|
|
|
if (ob.drop_hint) {
|
|
|
|
// That block was dropped by the data loader thread, but we were still expecting it...
|
|
|
|
// This is not good, because it means the loader is out of sync due to a bug.
|
|
|
|
// We can recover with the removal from `loading_blocks` so it will be re-queried again later...
|
|
|
|
print_line(String("Received a block loading drop while we were still expecting it: lod{0} ({1}, {2}, {3})")
|
|
|
|
.format(varray(ob.lod, ob.position.x, ob.position.y, ob.position.z)));
|
2019-05-20 12:49:46 -07:00
|
|
|
#ifdef TOOLS_ENABLED
|
|
|
|
lod.debug_unexpected_load_drop_time[ob.position] = OS::get_singleton()->get_ticks_msec();
|
|
|
|
#endif
|
2019-05-20 12:48:58 -07:00
|
|
|
++_stats.dropped_block_loads;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ob.data.voxels_loaded->get_size() != lod.map->get_block_size()) {
|
2019-05-03 16:02:10 -07:00
|
|
|
// Voxel block size is incorrect, drop it
|
2019-05-27 17:10:50 -07:00
|
|
|
ERR_PRINT("Block size obtained from stream is different from expected size");
|
2019-05-12 08:30:45 -07:00
|
|
|
++_stats.dropped_block_loads;
|
2019-05-03 16:02:10 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store buffer
|
2019-05-20 12:48:58 -07:00
|
|
|
VoxelBlock *block = lod.map->set_block_buffer(ob.position, ob.data.voxels_loaded);
|
2019-05-08 12:53:51 -07:00
|
|
|
//print_line(String("Adding block {0} at lod {1}").format(varray(eo.block_position.to_vec3(), eo.lod)));
|
2019-05-10 10:56:48 -07:00
|
|
|
// The block will be made visible and meshed only by LodOctree
|
2019-05-08 12:53:51 -07:00
|
|
|
block->set_visible(false);
|
2019-08-31 13:46:19 -07:00
|
|
|
block->set_parent_visible(is_visible());
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-04 17:09:12 -07:00
|
|
|
_stats.time_process_load_responses = profiling_clock.restart();
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
// Send mesh updates
|
|
|
|
{
|
|
|
|
VoxelMeshUpdater::Input input;
|
2019-05-12 08:33:25 -07:00
|
|
|
input.priority_position = viewer_block_pos;
|
2019-05-19 13:04:26 -07:00
|
|
|
input.priority_direction = viewer_direction;
|
2019-08-29 17:08:06 -07:00
|
|
|
input.use_exclusive_region = true;
|
|
|
|
input.exclusive_region_max_lod = get_lod_count() - 1;
|
2019-05-12 08:33:25 -07:00
|
|
|
input.exclusive_region_extent = get_block_region_extent();
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
for (unsigned int lod_index = 0; lod_index < get_lod_count(); ++lod_index) {
|
|
|
|
Lod &lod = _lods[lod_index];
|
|
|
|
|
2019-06-17 22:24:56 -07:00
|
|
|
for (unsigned int i = 0; i < lod.blocks_pending_update.size(); ++i) {
|
2019-05-03 16:02:10 -07:00
|
|
|
Vector3i block_pos = lod.blocks_pending_update[i];
|
|
|
|
|
2019-05-09 11:11:17 -07:00
|
|
|
VoxelBlock *block = lod.map->get_block(block_pos);
|
|
|
|
CRASH_COND(block == nullptr);
|
2019-05-09 14:00:54 -07:00
|
|
|
CRASH_COND(block->get_mesh_state() != VoxelBlock::MESH_UPDATE_NOT_SENT);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
// TODO Perhaps we could do a bit of early-rejection before spending time in buffer copy?
|
|
|
|
|
|
|
|
// Create buffer padded with neighbor voxels
|
|
|
|
Ref<VoxelBuffer> nbuffer;
|
|
|
|
nbuffer.instance();
|
|
|
|
|
|
|
|
// TODO Make the buffer re-usable
|
|
|
|
unsigned int block_size = lod.map->get_block_size();
|
|
|
|
unsigned int padding = _block_updater->get_required_padding();
|
|
|
|
nbuffer->create(
|
|
|
|
block_size + 2 * padding,
|
|
|
|
block_size + 2 * padding,
|
|
|
|
block_size + 2 * padding);
|
|
|
|
|
|
|
|
unsigned int channels_mask = (1 << VoxelBuffer::CHANNEL_ISOLEVEL);
|
|
|
|
lod.map->get_buffer_copy(lod.map->block_to_voxel(block_pos) - Vector3i(padding), **nbuffer, channels_mask);
|
|
|
|
|
|
|
|
VoxelMeshUpdater::InputBlock iblock;
|
2019-05-19 10:27:49 -07:00
|
|
|
iblock.data.voxels = nbuffer;
|
2019-05-03 16:02:10 -07:00
|
|
|
iblock.position = block_pos;
|
|
|
|
iblock.lod = lod_index;
|
|
|
|
input.blocks.push_back(iblock);
|
|
|
|
|
2019-05-09 14:00:54 -07:00
|
|
|
block->set_mesh_state(VoxelBlock::MESH_UPDATE_SENT);
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
lod.blocks_pending_update.clear();
|
|
|
|
}
|
|
|
|
|
2019-05-08 12:53:51 -07:00
|
|
|
//print_line(String("Sending {0} updates").format(varray(input.blocks.size())));
|
2019-05-03 16:02:10 -07:00
|
|
|
_block_updater->push(input);
|
|
|
|
}
|
|
|
|
|
2019-05-04 17:09:12 -07:00
|
|
|
_stats.time_request_blocks_to_update = profiling_clock.restart();
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
// Receive mesh updates
|
|
|
|
{
|
|
|
|
{
|
|
|
|
VoxelMeshUpdater::Output output;
|
|
|
|
_block_updater->pop(output);
|
2019-08-25 10:47:43 -07:00
|
|
|
|
2019-05-04 17:09:12 -07:00
|
|
|
_stats.updater = output.stats;
|
2019-08-25 10:47:43 -07:00
|
|
|
_stats.updated_blocks = output.blocks.size();
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-06-22 06:14:27 -07:00
|
|
|
for (int i = 0; i < output.blocks.size(); ++i) {
|
2019-05-03 16:02:10 -07:00
|
|
|
const VoxelMeshUpdater::OutputBlock &ob = output.blocks[i];
|
|
|
|
|
|
|
|
if (ob.lod >= get_lod_count()) {
|
|
|
|
// Sorry, LOD configuration changed, drop that mesh
|
2019-05-12 08:30:45 -07:00
|
|
|
++_stats.dropped_block_meshs;
|
2019-05-03 16:02:10 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
_blocks_pending_main_thread_update.push_back(ob);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ref<World> world = get_world();
|
2019-05-19 10:27:49 -07:00
|
|
|
uint32_t timeout = os.get_ticks_msec() + MAIN_THREAD_MESHING_BUDGET_MS; // Allocate milliseconds max to upload meshes
|
2019-06-17 22:24:56 -07:00
|
|
|
unsigned int queue_index = 0;
|
2019-05-03 16:02:10 -07:00
|
|
|
|
|
|
|
// The following is done on the main thread because Godot doesn't really support multithreaded Mesh allocation.
|
|
|
|
// This also proved to be very slow compared to the meshing process itself...
|
|
|
|
// hopefully Vulkan will allow us to upload graphical resources without stalling rendering as they upload?
|
|
|
|
|
|
|
|
for (; queue_index < _blocks_pending_main_thread_update.size() && os.get_ticks_msec() < timeout; ++queue_index) {
|
|
|
|
|
|
|
|
const VoxelMeshUpdater::OutputBlock &ob = _blocks_pending_main_thread_update[queue_index];
|
|
|
|
|
|
|
|
if (ob.lod >= get_lod_count()) {
|
|
|
|
// Sorry, LOD configuration changed, drop that mesh
|
2019-05-12 08:30:45 -07:00
|
|
|
++_stats.dropped_block_meshs;
|
2019-05-03 16:02:10 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Lod &lod = _lods[ob.lod];
|
|
|
|
|
|
|
|
VoxelBlock *block = lod.map->get_block(ob.position);
|
|
|
|
if (block == NULL) {
|
|
|
|
// That block is no longer loaded, drop the result
|
2019-05-12 08:30:45 -07:00
|
|
|
++_stats.dropped_block_meshs;
|
2019-05-03 16:02:10 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-05-20 12:48:58 -07:00
|
|
|
if (ob.drop_hint) {
|
|
|
|
// That block is loaded, but its meshing request was dropped.
|
|
|
|
// TODO Not sure what to do in this case, the code sending update queries has to be tweaked
|
|
|
|
print_line("Received a block mesh drop while we were still expecting it");
|
|
|
|
++_stats.dropped_block_meshs;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-05-09 14:00:54 -07:00
|
|
|
if (block->get_mesh_state() == VoxelBlock::MESH_UPDATE_SENT) {
|
|
|
|
block->set_mesh_state(VoxelBlock::MESH_UP_TO_DATE);
|
2019-05-09 11:11:17 -07:00
|
|
|
}
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
Ref<ArrayMesh> mesh;
|
|
|
|
mesh.instance();
|
|
|
|
|
2019-08-25 14:59:55 -07:00
|
|
|
// TODO Allow multiple collision surfaces
|
|
|
|
Array collidable_surface;
|
|
|
|
|
2019-06-17 22:24:56 -07:00
|
|
|
unsigned int surface_index = 0;
|
2019-05-19 10:27:49 -07:00
|
|
|
const VoxelMeshUpdater::OutputBlockData &data = ob.data;
|
|
|
|
for (int i = 0; i < data.smooth_surfaces.surfaces.size(); ++i) {
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-05-19 10:27:49 -07:00
|
|
|
Array surface = data.smooth_surfaces.surfaces[i];
|
2019-05-03 16:02:10 -07:00
|
|
|
if (surface.empty()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
CRASH_COND(surface.size() != Mesh::ARRAY_MAX);
|
2019-08-25 14:59:55 -07:00
|
|
|
if (!is_surface_triangulated(surface)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (collidable_surface.empty()) {
|
|
|
|
collidable_surface = surface;
|
|
|
|
}
|
|
|
|
|
2019-05-19 10:27:49 -07:00
|
|
|
mesh->add_surface_from_arrays(data.smooth_surfaces.primitive_type, surface);
|
2019-05-03 16:02:10 -07:00
|
|
|
mesh->surface_set_material(surface_index, _material);
|
|
|
|
// No multi-material supported yet
|
|
|
|
++surface_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_mesh_empty(mesh)) {
|
|
|
|
mesh = Ref<Mesh>();
|
|
|
|
}
|
|
|
|
|
2019-08-25 05:04:49 -07:00
|
|
|
bool has_collision = _generate_collisions;
|
|
|
|
if (has_collision && _collision_lod_count != -1) {
|
|
|
|
has_collision = ob.lod < _collision_lod_count;
|
|
|
|
}
|
|
|
|
|
2019-08-25 14:59:55 -07:00
|
|
|
block->set_mesh(mesh, this, has_collision, collidable_surface, get_tree()->is_debugging_collisions_hint());
|
2019-05-09 14:00:54 -07:00
|
|
|
block->mark_been_meshed();
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
shift_up(_blocks_pending_main_thread_update, queue_index);
|
|
|
|
}
|
|
|
|
|
2019-05-04 17:09:12 -07:00
|
|
|
_stats.time_process_update_responses = profiling_clock.restart();
|
|
|
|
}
|
|
|
|
|
2019-08-25 09:40:19 -07:00
|
|
|
Dictionary VoxelLodTerrain::get_statistics() const {
|
2019-05-04 17:09:12 -07:00
|
|
|
|
|
|
|
Dictionary d;
|
2019-05-27 17:10:50 -07:00
|
|
|
d["stream"] = VoxelDataLoader::Mgr::to_dictionary(_stats.stream);
|
2019-05-19 10:27:49 -07:00
|
|
|
d["updater"] = VoxelMeshUpdater::Mgr::to_dictionary(_stats.updater);
|
2019-08-25 10:47:43 -07:00
|
|
|
|
|
|
|
// Breakdown of time spent in _process
|
|
|
|
d["time_detect_required_blocks"] = _stats.time_detect_required_blocks;
|
|
|
|
d["time_request_blocks_to_load"] = _stats.time_request_blocks_to_load;
|
|
|
|
d["time_process_load_responses"] = _stats.time_process_load_responses;
|
|
|
|
d["time_request_blocks_to_update"] = _stats.time_request_blocks_to_update;
|
|
|
|
d["time_process_update_responses"] = _stats.time_process_update_responses;
|
|
|
|
|
|
|
|
d["remaining_main_thread_blocks"] = _blocks_pending_main_thread_update.size();
|
2019-05-12 08:30:45 -07:00
|
|
|
d["dropped_block_loads"] = _stats.dropped_block_loads;
|
|
|
|
d["dropped_block_meshs"] = _stats.dropped_block_meshs;
|
2019-08-25 10:47:43 -07:00
|
|
|
d["updated_blocks"] = _stats.updated_blocks;
|
|
|
|
d["blocked_lods"] = _stats.blocked_lods;
|
2019-05-04 17:09:12 -07:00
|
|
|
|
|
|
|
return d;
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::_bind_methods() {
|
|
|
|
|
2019-05-27 17:10:50 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("set_stream", "stream"), &VoxelLodTerrain::set_stream);
|
2019-05-25 08:07:38 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("get_stream"), &VoxelLodTerrain::get_stream);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-05-05 11:54:47 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("set_material", "material"), &VoxelLodTerrain::set_material);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_material"), &VoxelLodTerrain::get_material);
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("set_view_distance", "distance_in_voxels"), &VoxelLodTerrain::set_view_distance);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_view_distance"), &VoxelLodTerrain::get_view_distance);
|
|
|
|
|
2019-08-25 05:04:49 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("get_generate_collisions"), &VoxelLodTerrain::get_generate_collisions);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_generate_collisions", "enabled"), &VoxelLodTerrain::set_generate_collisions);
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_collision_lod_count"), &VoxelLodTerrain::get_collision_lod_count);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_collision_lod_count", "count"), &VoxelLodTerrain::set_collision_lod_count);
|
|
|
|
|
2019-05-03 16:02:10 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("get_viewer_path"), &VoxelLodTerrain::get_viewer_path);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_viewer_path", "path"), &VoxelLodTerrain::set_viewer_path);
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("set_lod_count", "lod_count"), &VoxelLodTerrain::set_lod_count);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_lod_count"), &VoxelLodTerrain::get_lod_count);
|
|
|
|
|
2019-05-05 11:54:47 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("set_lod_split_scale", "lod_split_scale"), &VoxelLodTerrain::set_lod_split_scale);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_lod_split_scale"), &VoxelLodTerrain::get_lod_split_scale);
|
|
|
|
|
2019-05-07 16:57:34 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("get_block_region_extent"), &VoxelLodTerrain::get_block_region_extent);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_block_info", "block_pos", "lod"), &VoxelLodTerrain::get_block_info);
|
2019-08-25 09:40:19 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("get_statistics"), &VoxelLodTerrain::get_statistics);
|
2019-05-09 16:57:09 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("voxel_to_block_position", "lod_index"), &VoxelLodTerrain::voxel_to_block_position);
|
2019-05-03 16:02:10 -07:00
|
|
|
|
2019-08-23 17:44:27 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("_on_stream_params_changed"), &VoxelLodTerrain::_on_stream_params_changed);
|
|
|
|
|
2019-05-27 17:10:50 -07:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "stream", PROPERTY_HINT_RESOURCE_TYPE, "VoxelStream"), "set_stream", "get_stream");
|
2019-05-03 16:02:10 -07:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "view_distance"), "set_view_distance", "get_view_distance");
|
2019-05-05 11:54:47 -07:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "lod_count"), "set_lod_count", "get_lod_count");
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::REAL, "lod_split_scale"), "set_lod_split_scale", "get_lod_split_scale");
|
2019-05-03 16:02:10 -07:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::NODE_PATH, "viewer_path"), "set_viewer_path", "get_viewer_path");
|
2019-05-05 11:54:47 -07:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "material", PROPERTY_HINT_RESOURCE_TYPE, "Material"), "set_material", "get_material");
|
2019-08-25 05:04:49 -07:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "generate_collisions"), "set_generate_collisions", "get_generate_collisions");
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "collision_lod_count"), "set_collision_lod_count", "get_collision_lod_count");
|
2019-05-03 16:02:10 -07:00
|
|
|
}
|
2019-08-29 14:55:02 -07:00
|
|
|
|
|
|
|
#ifdef TOOLS_ENABLED
|
|
|
|
|
|
|
|
void VoxelLodTerrain::create_octree_debug_box(OctreeItem &item, Vector3i pos) {
|
|
|
|
CRASH_COND(item.debug_box != nullptr);
|
|
|
|
MeshInstance *mi = memnew(MeshInstance);
|
|
|
|
mi->set_mesh(VoxelDebug::get_debug_box_mesh());
|
|
|
|
float s = 1 << (get_block_size_pow2() + get_lod_count() - 1);
|
|
|
|
mi->set_scale(Vector3(s, s, s));
|
|
|
|
mi->set_translation(pos.to_vec3() * s);
|
|
|
|
add_child(mi);
|
|
|
|
item.debug_box = mi;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelLodTerrain::destroy_octree_debug_box(OctreeItem &item, Vector3i pos) {
|
|
|
|
if (item.debug_box == nullptr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
item.debug_box->queue_delete();
|
|
|
|
item.debug_box = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|