2020-01-31 19:52:37 +01:00
|
|
|
/*
|
|
|
|
Copyright (c) 2019-2020 Péter Magyar
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2019-06-07 01:33:41 +02:00
|
|
|
#ifndef VOXEL_WORLD_H
|
|
|
|
#define VOXEL_WORLD_H
|
|
|
|
|
2019-11-19 16:33:06 +01:00
|
|
|
#include "core/engine.h"
|
2020-01-09 04:29:05 +01:00
|
|
|
#include "core/hash_map.h"
|
|
|
|
#include "scene/3d/navigation.h"
|
2019-06-07 01:33:41 +02:00
|
|
|
|
2019-11-10 01:03:48 +01:00
|
|
|
#include "../areas/world_area.h"
|
2020-01-09 04:29:05 +01:00
|
|
|
#include "../level_generator/voxelman_level_generator.h"
|
|
|
|
#include "../library/voxelman_library.h"
|
2019-06-07 01:33:41 +02:00
|
|
|
|
2019-11-19 14:42:21 +01:00
|
|
|
#include "core/os/os.h"
|
|
|
|
|
2019-08-12 20:40:05 +02:00
|
|
|
class VoxelChunk;
|
|
|
|
|
2019-10-05 22:53:48 +02:00
|
|
|
class VoxelWorld : public Navigation {
|
|
|
|
GDCLASS(VoxelWorld, Navigation);
|
2019-06-07 01:33:41 +02:00
|
|
|
|
|
|
|
public:
|
2020-01-09 04:29:05 +01:00
|
|
|
int get_chunk_size_x() const;
|
|
|
|
void set_chunk_size_x(const int value);
|
|
|
|
|
|
|
|
int get_chunk_size_y() const;
|
|
|
|
void set_chunk_size_y(const int value);
|
|
|
|
|
|
|
|
int get_chunk_size_z() const;
|
|
|
|
void set_chunk_size_z(const int value);
|
|
|
|
|
2020-02-14 03:19:15 +01:00
|
|
|
int get_data_margin_start() const;
|
|
|
|
void set_data_margin_start(const int value);
|
|
|
|
|
|
|
|
int get_data_margin_end() const;
|
|
|
|
void set_data_margin_end(const int value);
|
|
|
|
|
2020-01-09 04:29:05 +01:00
|
|
|
int get_current_seed() const;
|
|
|
|
void set_current_seed(const int value);
|
2019-11-19 14:42:21 +01:00
|
|
|
|
|
|
|
bool get_use_threads();
|
|
|
|
void set_use_threads(bool value);
|
|
|
|
|
|
|
|
uint32_t get_max_concurrent_generations();
|
|
|
|
void set_max_concurrent_generations(uint32_t value);
|
2020-01-09 04:29:05 +01:00
|
|
|
|
|
|
|
Ref<VoxelmanLibrary> get_library() const;
|
|
|
|
void set_library(const Ref<VoxelmanLibrary> library);
|
|
|
|
|
|
|
|
Ref<VoxelmanLevelGenerator> get_level_generator() const;
|
|
|
|
void set_level_generator(const Ref<VoxelmanLevelGenerator> level_generator);
|
2019-08-12 20:40:05 +02:00
|
|
|
|
|
|
|
float get_voxel_scale() const;
|
|
|
|
void set_voxel_scale(const float value);
|
|
|
|
|
|
|
|
int get_chunk_spawn_range() const;
|
|
|
|
void set_chunk_spawn_range(const int value);
|
|
|
|
|
2019-06-07 01:33:41 +02:00
|
|
|
NodePath get_player_path();
|
|
|
|
void set_player_path(NodePath player_path);
|
2020-01-09 04:29:05 +01:00
|
|
|
|
|
|
|
Spatial *get_player() const;
|
2019-08-12 20:40:05 +02:00
|
|
|
void set_player(Spatial *player);
|
|
|
|
void set_player_bind(Node *player);
|
|
|
|
|
2019-11-10 01:03:48 +01:00
|
|
|
Ref<WorldArea> get_world_area(const int index) const;
|
|
|
|
void add_world_area(Ref<WorldArea> area);
|
|
|
|
void remove_world_area(const int index);
|
|
|
|
void clear_world_areas();
|
|
|
|
int get_world_area_count() const;
|
|
|
|
|
2019-11-06 03:37:22 +01:00
|
|
|
void add_chunk(VoxelChunk *chunk, const int x, const int y, const int z);
|
|
|
|
void add_chunk_bind(Node *chunk, const int x, const int y, const int z);
|
|
|
|
VoxelChunk *get_chunk(const int x, const int y, const int z) const;
|
|
|
|
VoxelChunk *remove_chunk(const int x, const int y, const int z);
|
2019-08-12 20:40:05 +02:00
|
|
|
|
2019-11-06 03:37:22 +01:00
|
|
|
VoxelChunk *get_chunk_index(const int index);
|
2019-08-12 20:40:05 +02:00
|
|
|
int get_chunk_count() const;
|
|
|
|
|
2019-11-19 16:33:06 +01:00
|
|
|
void add_to_generation_queue_bind(Node *chunk);
|
|
|
|
void add_to_generation_queue(VoxelChunk *chunk);
|
|
|
|
VoxelChunk *get_generation_queue_index(int index);
|
|
|
|
void remove_generation_queue_index(int index);
|
|
|
|
int get_generation_queue_size();
|
2019-06-07 01:33:41 +02:00
|
|
|
|
2019-11-19 16:33:06 +01:00
|
|
|
void add_to_generation_bind(Node *chunk);
|
|
|
|
void add_to_generation(VoxelChunk *chunk);
|
|
|
|
VoxelChunk *get_generation_index(int index);
|
|
|
|
void remove_generation_index(int index);
|
|
|
|
int get_generation_size();
|
|
|
|
|
|
|
|
void clear();
|
|
|
|
|
|
|
|
VoxelChunk *create_chunk(int x, int y, int z);
|
2019-11-19 20:39:45 +01:00
|
|
|
VoxelChunk *_create_chunk(int x, int y, int z, Node *p_chunk);
|
2019-11-19 14:42:21 +01:00
|
|
|
|
|
|
|
void generate_chunk_bind(Node *p_chunk);
|
|
|
|
void generate_chunk(VoxelChunk *p_chunk);
|
|
|
|
void _generate_chunk(Node *p_chunk);
|
|
|
|
|
2019-08-12 13:12:42 +02:00
|
|
|
VoxelWorld();
|
2019-06-07 19:13:07 +02:00
|
|
|
~VoxelWorld();
|
2019-06-07 01:33:41 +02:00
|
|
|
|
|
|
|
protected:
|
2019-11-19 14:42:21 +01:00
|
|
|
virtual void _notification(int p_what);
|
2019-06-07 01:33:41 +02:00
|
|
|
static void _bind_methods();
|
|
|
|
|
2019-11-18 22:22:41 +01:00
|
|
|
public:
|
2019-11-18 21:53:33 +01:00
|
|
|
struct IntPos {
|
|
|
|
int x;
|
|
|
|
int y;
|
|
|
|
int z;
|
|
|
|
|
|
|
|
IntPos() {
|
|
|
|
x = 0;
|
|
|
|
y = 0;
|
|
|
|
z = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
IntPos(int p_x, int p_y, int p_z) {
|
|
|
|
x = p_x;
|
|
|
|
y = p_y;
|
|
|
|
z = p_z;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct IntPosHasher {
|
|
|
|
static _FORCE_INLINE_ uint32_t hash(const IntPos &v) {
|
|
|
|
uint32_t hash = hash_djb2_one_32(v.x);
|
|
|
|
hash = hash_djb2_one_32(v.y, hash);
|
|
|
|
return hash_djb2_one_32(v.z, hash);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-06-07 01:33:41 +02:00
|
|
|
private:
|
2020-01-09 04:29:05 +01:00
|
|
|
int _chunk_size_x;
|
2019-11-18 21:53:33 +01:00
|
|
|
int _chunk_size_y;
|
|
|
|
int _chunk_size_z;
|
2019-11-19 14:42:21 +01:00
|
|
|
int _current_seed;
|
2020-02-14 03:19:15 +01:00
|
|
|
int _data_margin_start;
|
|
|
|
int _data_margin_end;
|
2020-01-09 04:29:05 +01:00
|
|
|
|
|
|
|
Ref<VoxelmanLibrary> _library;
|
|
|
|
Ref<VoxelmanLevelGenerator> _level_generator;
|
2019-08-12 20:40:05 +02:00
|
|
|
float _voxel_scale;
|
|
|
|
int _chunk_spawn_range;
|
2020-01-09 04:29:05 +01:00
|
|
|
|
|
|
|
HashMap<IntPos, VoxelChunk *, IntPosHasher> _chunks;
|
|
|
|
Vector<VoxelChunk *> _chunks_vector;
|
2019-06-07 01:33:41 +02:00
|
|
|
|
2019-11-10 01:03:48 +01:00
|
|
|
Vector<Ref<WorldArea> > _world_areas;
|
|
|
|
|
2019-06-07 01:33:41 +02:00
|
|
|
NodePath _player_path;
|
|
|
|
Spatial *_player;
|
2019-11-19 14:42:21 +01:00
|
|
|
|
|
|
|
bool _use_threads;
|
|
|
|
uint32_t _max_concurrent_generations;
|
|
|
|
Vector<VoxelChunk *> _generation_queue;
|
|
|
|
Vector<VoxelChunk *> _generating;
|
2019-06-07 01:33:41 +02:00
|
|
|
};
|
|
|
|
|
2019-11-18 22:22:41 +01:00
|
|
|
_FORCE_INLINE_ bool operator==(const VoxelWorld::IntPos &a, const VoxelWorld::IntPos &b) {
|
|
|
|
return a.x == b.x && a.y == b.y && a.z == b.z;
|
|
|
|
}
|
|
|
|
|
2019-06-07 01:33:41 +02:00
|
|
|
#endif
|