2016-05-10 01:59:54 +02:00
|
|
|
#ifndef VOXEL_TERRAIN_H
|
|
|
|
#define VOXEL_TERRAIN_H
|
|
|
|
|
2019-04-28 17:58:29 +01:00
|
|
|
#include "../math/rect3i.h"
|
|
|
|
#include "../math/vector3i.h"
|
|
|
|
#include "../util/zprofiling.h"
|
2019-05-25 15:07:16 +01:00
|
|
|
#include "voxel_data_loader.h"
|
2019-04-24 01:29:47 +01:00
|
|
|
#include "voxel_mesh_updater.h"
|
2018-09-25 00:54:07 +01:00
|
|
|
|
|
|
|
#include <scene/3d/spatial.h>
|
|
|
|
|
|
|
|
class VoxelMap;
|
|
|
|
class VoxelLibrary;
|
2019-09-05 19:43:25 +01:00
|
|
|
class VoxelStream;
|
2019-09-08 19:42:25 +01:00
|
|
|
class VoxelTool;
|
2017-03-26 18:09:37 +02:00
|
|
|
|
2019-05-09 19:11:17 +01:00
|
|
|
// Infinite paged terrain made of voxel blocks all with the same level of detail.
|
2019-05-04 00:00:50 +01:00
|
|
|
// Voxels are polygonized around the viewer by distance in a large cubic space.
|
2019-05-25 16:07:38 +01:00
|
|
|
// Data is streamed using a VoxelStream.
|
2019-04-24 01:29:47 +01:00
|
|
|
class VoxelTerrain : public Spatial {
|
2017-08-20 18:37:08 +02:00
|
|
|
GDCLASS(VoxelTerrain, Spatial)
|
2016-05-10 01:59:54 +02:00
|
|
|
public:
|
2017-01-01 04:40:16 +01:00
|
|
|
VoxelTerrain();
|
2018-09-25 00:54:07 +01:00
|
|
|
~VoxelTerrain();
|
2016-05-10 01:59:54 +02:00
|
|
|
|
2019-08-24 01:44:27 +01:00
|
|
|
void set_stream(Ref<VoxelStream> p_stream);
|
2019-05-25 16:07:38 +01:00
|
|
|
Ref<VoxelStream> get_stream() const;
|
2017-08-15 02:24:52 +02:00
|
|
|
|
2019-08-24 01:44:27 +01:00
|
|
|
unsigned int get_block_size_pow2() const;
|
|
|
|
void set_block_size_po2(unsigned int p_block_size_po2);
|
|
|
|
|
2017-08-15 02:24:52 +02:00
|
|
|
void set_voxel_library(Ref<VoxelLibrary> library);
|
|
|
|
Ref<VoxelLibrary> get_voxel_library() const;
|
2016-05-10 01:59:54 +02:00
|
|
|
|
2017-03-28 00:56:01 +02:00
|
|
|
void make_block_dirty(Vector3i bpos);
|
2018-09-25 00:54:07 +01:00
|
|
|
//void make_blocks_dirty(Vector3i min, Vector3i size);
|
2017-03-29 22:36:42 +02:00
|
|
|
void make_voxel_dirty(Vector3i pos);
|
2018-10-01 20:48:47 +01:00
|
|
|
void make_area_dirty(Rect3i box);
|
2016-05-10 01:59:54 +02:00
|
|
|
|
2017-03-26 18:09:37 +02:00
|
|
|
void set_generate_collisions(bool enabled);
|
2017-08-15 02:24:52 +02:00
|
|
|
bool get_generate_collisions() const { return _generate_collisions; }
|
|
|
|
|
|
|
|
int get_view_distance() const;
|
|
|
|
void set_view_distance(int distance_in_voxels);
|
2017-03-26 18:09:37 +02:00
|
|
|
|
2017-03-28 00:56:01 +02:00
|
|
|
void set_viewer_path(NodePath path);
|
2017-08-15 02:24:52 +02:00
|
|
|
NodePath get_viewer_path() const;
|
|
|
|
|
2019-06-18 14:24:56 +09:00
|
|
|
void set_material(unsigned int id, Ref<Material> material);
|
|
|
|
Ref<Material> get_material(unsigned int id) const;
|
2017-03-28 00:56:01 +02:00
|
|
|
|
2019-04-28 01:32:23 +01:00
|
|
|
bool is_smooth_meshing_enabled() const;
|
|
|
|
void set_smooth_meshing_enabled(bool enabled);
|
|
|
|
|
2019-09-03 22:54:40 +01:00
|
|
|
Ref<VoxelMap> get_storage() { return _map; }
|
2019-09-08 19:42:25 +01:00
|
|
|
Ref<VoxelTool> get_voxel_tool();
|
2016-05-10 01:59:54 +02:00
|
|
|
|
2018-09-25 00:54:07 +01:00
|
|
|
struct Stats {
|
|
|
|
VoxelMeshUpdater::Stats updater;
|
2019-05-28 01:10:50 +01:00
|
|
|
VoxelDataLoader::Stats stream;
|
2019-08-25 18:47:43 +01:00
|
|
|
int updated_blocks = 0;
|
|
|
|
int dropped_block_loads = 0;
|
|
|
|
int dropped_block_meshs = 0;
|
|
|
|
uint64_t time_detect_required_blocks = 0;
|
|
|
|
uint64_t time_request_blocks_to_load = 0;
|
|
|
|
uint64_t time_process_load_responses = 0;
|
|
|
|
uint64_t time_request_blocks_to_update = 0;
|
|
|
|
uint64_t time_process_update_responses = 0;
|
2018-09-25 00:54:07 +01:00
|
|
|
};
|
|
|
|
|
2016-05-10 01:59:54 +02:00
|
|
|
protected:
|
2017-01-01 04:40:16 +01:00
|
|
|
void _notification(int p_what);
|
2017-03-28 00:56:01 +02:00
|
|
|
|
|
|
|
private:
|
2017-08-15 02:24:52 +02:00
|
|
|
bool _set(const StringName &p_name, const Variant &p_value);
|
|
|
|
bool _get(const StringName &p_name, Variant &r_ret) const;
|
|
|
|
void _get_property_list(List<PropertyInfo> *p_list) const;
|
|
|
|
|
2017-01-01 04:40:16 +01:00
|
|
|
void _process();
|
2016-05-10 01:59:54 +02:00
|
|
|
|
2019-08-24 01:44:27 +01:00
|
|
|
void _on_stream_params_changed();
|
|
|
|
void _set_block_size_po2(int p_block_size_po2);
|
2017-08-28 01:47:38 +02:00
|
|
|
void make_all_view_dirty_deferred();
|
2019-08-24 01:44:27 +01:00
|
|
|
void start_updater();
|
|
|
|
void stop_updater();
|
|
|
|
void start_streamer();
|
|
|
|
void stop_streamer();
|
|
|
|
void reset_map();
|
2017-08-15 02:24:52 +02:00
|
|
|
|
2019-09-07 21:19:12 +01:00
|
|
|
Spatial *get_viewer() const;
|
2017-03-28 00:56:01 +02:00
|
|
|
|
2017-08-28 01:47:38 +02:00
|
|
|
void immerge_block(Vector3i bpos);
|
2019-05-28 00:40:09 +01:00
|
|
|
void save_all_modified_blocks(bool with_copy);
|
2019-09-07 21:19:12 +01:00
|
|
|
void get_viewer_pos_and_direction(Vector3 &out_pos, Vector3 &out_direction) const;
|
2019-06-02 01:59:39 +01:00
|
|
|
void send_block_data_requests();
|
2017-08-28 01:47:38 +02:00
|
|
|
|
2018-09-25 00:54:07 +01:00
|
|
|
Dictionary get_statistics() const;
|
2016-05-10 01:59:54 +02:00
|
|
|
|
2017-01-01 04:40:16 +01:00
|
|
|
static void _bind_methods();
|
2016-05-10 01:59:54 +02:00
|
|
|
|
2019-09-08 19:42:25 +01:00
|
|
|
// Bindings
|
|
|
|
Vector3 _b_voxel_to_block(Vector3 pos);
|
|
|
|
Vector3 _b_block_to_voxel(Vector3 pos);
|
2018-09-25 00:54:07 +01:00
|
|
|
//void _force_load_blocks_binding(Vector3 center, Vector3 extents) { force_load_blocks(center, extents); }
|
|
|
|
|
2017-01-02 02:19:02 +01:00
|
|
|
private:
|
|
|
|
// Voxel storage
|
|
|
|
Ref<VoxelMap> _map;
|
|
|
|
|
2017-08-15 02:24:52 +02:00
|
|
|
// How many blocks to load around the viewer
|
|
|
|
int _view_distance_blocks;
|
|
|
|
|
2017-03-28 00:56:01 +02:00
|
|
|
// TODO Terrains only need to handle the visible portion of voxels, which reduces the bounds blocks to handle.
|
|
|
|
// Therefore, could a simple grid be better to use than a hashmap?
|
|
|
|
|
2019-08-25 15:14:10 +01:00
|
|
|
Set<Vector3i> _loading_blocks;
|
2018-09-25 00:54:07 +01:00
|
|
|
Vector<Vector3i> _blocks_pending_load;
|
|
|
|
Vector<Vector3i> _blocks_pending_update;
|
2018-09-26 01:22:23 +01:00
|
|
|
Vector<VoxelMeshUpdater::OutputBlock> _blocks_pending_main_thread_update;
|
2017-04-06 23:44:11 +02:00
|
|
|
|
2019-05-28 00:40:09 +01:00
|
|
|
std::vector<VoxelDataLoader::InputBlock> _blocks_to_save;
|
|
|
|
|
2019-05-25 16:07:38 +01:00
|
|
|
Ref<VoxelStream> _stream;
|
|
|
|
VoxelDataLoader *_stream_thread;
|
2018-09-25 00:54:07 +01:00
|
|
|
|
|
|
|
Ref<VoxelLibrary> _library;
|
|
|
|
VoxelMeshUpdater *_block_updater;
|
2017-01-02 02:19:02 +01:00
|
|
|
|
2017-03-28 00:56:01 +02:00
|
|
|
NodePath _viewer_path;
|
2017-08-28 01:47:38 +02:00
|
|
|
Vector3i _last_viewer_block_pos;
|
|
|
|
int _last_view_distance_blocks;
|
2017-03-28 00:56:01 +02:00
|
|
|
|
2019-08-25 13:04:49 +01:00
|
|
|
bool _generate_collisions = true;
|
2018-09-28 01:10:35 +01:00
|
|
|
bool _run_in_editor;
|
2019-04-28 01:32:23 +01:00
|
|
|
bool _smooth_meshing_enabled;
|
2017-03-26 18:09:37 +02:00
|
|
|
|
2019-04-28 17:58:29 +01:00
|
|
|
Ref<Material> _materials[VoxelMesherBlocky::MAX_MATERIALS];
|
2018-09-25 00:54:07 +01:00
|
|
|
|
|
|
|
Stats _stats;
|
2016-05-10 01:59:54 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif // VOXEL_TERRAIN_H
|