mirror of
https://github.com/Relintai/pandemonium_engine.git
synced 2025-01-03 09:29:38 +01:00
Rework NavigationMeshInstance as per the navigation mesh generators pr.
This commit is contained in:
parent
f1f5849aff
commit
a1b23406a1
@ -30,13 +30,16 @@
|
|||||||
|
|
||||||
#include "navigation_mesh_instance.h"
|
#include "navigation_mesh_instance.h"
|
||||||
|
|
||||||
|
#include "core/core_string_names.h"
|
||||||
#include "core/os/os.h"
|
#include "core/os/os.h"
|
||||||
#include "mesh_instance.h"
|
#include "mesh_instance.h"
|
||||||
#include "navigation.h"
|
#include "navigation.h"
|
||||||
#include "scene/resources/mesh.h"
|
#include "scene/resources/mesh.h"
|
||||||
#include "scene/resources/navigation_mesh.h"
|
#include "scene/resources/navigation_mesh.h"
|
||||||
#include "scene/resources/world_3d.h"
|
#include "scene/resources/world_3d.h"
|
||||||
|
#include "servers/navigation/navigation_mesh_generator.h"
|
||||||
#include "servers/navigation_server.h"
|
#include "servers/navigation_server.h"
|
||||||
|
#include "scene/resources/navigation_mesh_source_geometry_data_3d.h"
|
||||||
|
|
||||||
void NavigationMeshInstance::set_enabled(bool p_enabled) {
|
void NavigationMeshInstance::set_enabled(bool p_enabled) {
|
||||||
if (enabled == p_enabled) {
|
if (enabled == p_enabled) {
|
||||||
@ -53,10 +56,14 @@ void NavigationMeshInstance::set_enabled(bool p_enabled) {
|
|||||||
} else {
|
} else {
|
||||||
if (navigation) {
|
if (navigation) {
|
||||||
NavigationServer::get_singleton()->region_set_map(region, navigation->get_rid());
|
NavigationServer::get_singleton()->region_set_map(region, navigation->get_rid());
|
||||||
|
} else {
|
||||||
|
if (map_override.is_valid()) {
|
||||||
|
NavigationServer::get_singleton()->region_set_map(region, map_override);
|
||||||
} else {
|
} else {
|
||||||
NavigationServer::get_singleton()->region_set_map(region, get_world_3d()->get_navigation_map());
|
NavigationServer::get_singleton()->region_set_map(region, get_world_3d()->get_navigation_map());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef DEBUG_ENABLED
|
#ifdef DEBUG_ENABLED
|
||||||
if (debug_instance.is_valid()) {
|
if (debug_instance.is_valid()) {
|
||||||
@ -142,8 +149,12 @@ void NavigationMeshInstance::_notification(int p_what) {
|
|||||||
|
|
||||||
if (enabled && navigation == nullptr) {
|
if (enabled && navigation == nullptr) {
|
||||||
// did not find a valid navigation node parent, fallback to default navigation map on world resource
|
// did not find a valid navigation node parent, fallback to default navigation map on world resource
|
||||||
|
if (map_override.is_valid()) {
|
||||||
|
NavigationServer::get_singleton()->region_set_map(region, map_override);
|
||||||
|
} else {
|
||||||
NavigationServer::get_singleton()->region_set_map(region, get_world_3d()->get_navigation_map());
|
NavigationServer::get_singleton()->region_set_map(region, get_world_3d()->get_navigation_map());
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef DEBUG_ENABLED
|
#ifdef DEBUG_ENABLED
|
||||||
if (NavigationServer::get_singleton()->get_debug_enabled()) {
|
if (NavigationServer::get_singleton()->get_debug_enabled()) {
|
||||||
@ -163,9 +174,7 @@ void NavigationMeshInstance::_notification(int p_what) {
|
|||||||
|
|
||||||
} break;
|
} break;
|
||||||
case NOTIFICATION_EXIT_TREE: {
|
case NOTIFICATION_EXIT_TREE: {
|
||||||
if (navigation) {
|
|
||||||
NavigationServer::get_singleton()->region_set_map(region, RID());
|
NavigationServer::get_singleton()->region_set_map(region, RID());
|
||||||
}
|
|
||||||
|
|
||||||
navigation = nullptr;
|
navigation = nullptr;
|
||||||
|
|
||||||
@ -182,22 +191,80 @@ void NavigationMeshInstance::_notification(int p_what) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void NavigationMeshInstance::set_navigation_mesh(const Ref<NavigationMesh> &p_navmesh) {
|
void NavigationMeshInstance::set_navigation_mesh(const Ref<NavigationMesh> &p_navigation_mesh) {
|
||||||
if (p_navmesh == navmesh) {
|
if (navmesh.is_valid()) {
|
||||||
|
navmesh->disconnect(CoreStringNames::get_singleton()->changed, this, "_navigation_mesh_changed");
|
||||||
|
}
|
||||||
|
|
||||||
|
navmesh = p_navigation_mesh;
|
||||||
|
|
||||||
|
if (navmesh.is_valid()) {
|
||||||
|
navmesh->connect(CoreStringNames::get_singleton()->changed, this, "_navigation_mesh_changed");
|
||||||
|
}
|
||||||
|
|
||||||
|
_navigation_mesh_changed();
|
||||||
|
}
|
||||||
|
|
||||||
|
Ref<NavigationMesh> NavigationMeshInstance::get_navigation_mesh() const {
|
||||||
|
return navmesh;
|
||||||
|
}
|
||||||
|
|
||||||
|
void NavigationMeshInstance::set_navigation_map(RID p_navigation_map) {
|
||||||
|
if (map_override == p_navigation_map) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (navmesh.is_valid()) {
|
map_override = p_navigation_map;
|
||||||
navmesh->remove_change_receptor(this);
|
|
||||||
|
NavigationServer::get_singleton()->region_set_map(region, map_override);
|
||||||
}
|
}
|
||||||
|
|
||||||
navmesh = p_navmesh;
|
RID NavigationMeshInstance::get_navigation_map() const {
|
||||||
|
if (map_override.is_valid()) {
|
||||||
if (navmesh.is_valid()) {
|
return map_override;
|
||||||
navmesh->add_change_receptor(this);
|
} else if (is_inside_tree()) {
|
||||||
|
return get_world_3d()->get_navigation_map();
|
||||||
|
}
|
||||||
|
return RID();
|
||||||
}
|
}
|
||||||
|
|
||||||
NavigationServer::get_singleton()->region_set_navmesh(region, p_navmesh);
|
void NavigationMeshInstance::bake_navigation_mesh(bool p_on_thread) {
|
||||||
|
ERR_FAIL_COND_MSG(!get_navigation_mesh().is_valid(), "Can't bake the navigation mesh if the `NavigationMesh` resource doesn't exist");
|
||||||
|
if (baking_started) {
|
||||||
|
WARN_PRINT("NavigationMesh baking already started. Wait for it to finish.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
baking_started = true;
|
||||||
|
|
||||||
|
navmesh->clear();
|
||||||
|
|
||||||
|
if (p_on_thread && OS::get_singleton()->get_render_thread_mode() == OS::RenderThreadMode::RENDER_SEPARATE_THREAD) {
|
||||||
|
Ref<FuncRef> f;
|
||||||
|
f.instance();
|
||||||
|
f->set_instance(this);
|
||||||
|
f->set_function("_bake_finished");
|
||||||
|
|
||||||
|
NavigationMeshGenerator::get_singleton()->parse_and_bake_3d(navmesh, this, f);
|
||||||
|
} else {
|
||||||
|
Ref<NavigationMeshSourceGeometryData3D> source_geometry_data = NavigationMeshGenerator::get_singleton()->parse_3d_source_geometry_data(navmesh, this);
|
||||||
|
NavigationMeshGenerator::get_singleton()->bake_3d_from_source_geometry_data(navmesh, source_geometry_data);
|
||||||
|
_bake_finished(navmesh);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void NavigationMeshInstance::_bake_finished(Ref<NavigationMesh> p_nav_mesh) {
|
||||||
|
baking_started = false;
|
||||||
|
set_navigation_mesh(p_nav_mesh);
|
||||||
|
emit_signal("bake_finished");
|
||||||
|
}
|
||||||
|
|
||||||
|
void NavigationMeshInstance::_navigation_mesh_changed() {
|
||||||
|
NavigationServer::get_singleton()->region_set_navmesh(region, navmesh);
|
||||||
|
|
||||||
|
update_gizmos();
|
||||||
|
update_configuration_warning();
|
||||||
|
|
||||||
|
emit_signal("navigation_mesh_changed");
|
||||||
|
|
||||||
#ifdef DEBUG_ENABLED
|
#ifdef DEBUG_ENABLED
|
||||||
if (is_inside_tree() && NavigationServer::get_singleton()->get_debug_enabled()) {
|
if (is_inside_tree() && NavigationServer::get_singleton()->get_debug_enabled()) {
|
||||||
@ -214,72 +281,6 @@ void NavigationMeshInstance::set_navigation_mesh(const Ref<NavigationMesh> &p_na
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif // DEBUG_ENABLED
|
#endif // DEBUG_ENABLED
|
||||||
|
|
||||||
emit_signal("navigation_mesh_changed");
|
|
||||||
|
|
||||||
update_gizmos();
|
|
||||||
update_configuration_warning();
|
|
||||||
}
|
|
||||||
|
|
||||||
Ref<NavigationMesh> NavigationMeshInstance::get_navigation_mesh() const {
|
|
||||||
return navmesh;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct BakeThreadsArgs {
|
|
||||||
NavigationMeshInstance *nav_region;
|
|
||||||
|
|
||||||
BakeThreadsArgs() {
|
|
||||||
nav_region = nullptr;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
void _bake_navigation_mesh(void *p_user_data) {
|
|
||||||
BakeThreadsArgs *args = static_cast<BakeThreadsArgs *>(p_user_data);
|
|
||||||
|
|
||||||
if (args->nav_region->get_navigation_mesh().is_valid()) {
|
|
||||||
Ref<NavigationMesh> nav_mesh = args->nav_region->get_navigation_mesh()->duplicate();
|
|
||||||
|
|
||||||
NavigationServer::get_singleton()->region_bake_navmesh(nav_mesh, args->nav_region);
|
|
||||||
args->nav_region->call_deferred("_bake_finished", nav_mesh);
|
|
||||||
memdelete(args);
|
|
||||||
} else {
|
|
||||||
ERR_PRINT("Can't bake the navigation mesh if the `NavigationMesh` resource doesn't exist");
|
|
||||||
args->nav_region->call_deferred("_bake_finished", Ref<NavigationMesh>());
|
|
||||||
memdelete(args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void NavigationMeshInstance::bake_navigation_mesh(bool p_on_thread) {
|
|
||||||
ERR_FAIL_COND_MSG(bake_thread.is_started(), "Navigation Mesh Bake thread is already baking a Navigation Mesh. Unable to start another bake request.");
|
|
||||||
|
|
||||||
BakeThreadsArgs *args = memnew(BakeThreadsArgs);
|
|
||||||
args->nav_region = this;
|
|
||||||
|
|
||||||
if (p_on_thread && !OS::get_singleton()->can_use_threads()) {
|
|
||||||
WARN_PRINT("NavigationMesh bake 'on_thread' will be disabled as the current OS does not support multiple threads."
|
|
||||||
"\nAs a fallback the navigation mesh will bake on the main thread which can cause framerate issues.");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (p_on_thread && OS::get_singleton()->can_use_threads()) {
|
|
||||||
bake_thread.start(_bake_navigation_mesh, args);
|
|
||||||
} else {
|
|
||||||
_bake_navigation_mesh(args);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void NavigationMeshInstance::_bake_finished(Ref<NavigationMesh> p_nav_mesh) {
|
|
||||||
set_navigation_mesh(p_nav_mesh);
|
|
||||||
bake_thread.wait_to_finish();
|
|
||||||
emit_signal("bake_finished");
|
|
||||||
}
|
|
||||||
|
|
||||||
void NavigationMeshInstance::_navigation_changed() {
|
|
||||||
update_gizmos();
|
|
||||||
update_configuration_warning();
|
|
||||||
|
|
||||||
#ifdef DEBUG_ENABLED
|
|
||||||
_update_debug_edge_connections_mesh();
|
|
||||||
#endif // DEBUG_ENABLED
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG_ENABLED
|
#ifdef DEBUG_ENABLED
|
||||||
@ -303,26 +304,29 @@ String NavigationMeshInstance::get_configuration_warning() const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void NavigationMeshInstance::_bind_methods() {
|
void NavigationMeshInstance::_bind_methods() {
|
||||||
ClassDB::bind_method(D_METHOD("set_navigation_mesh", "navmesh"), &NavigationMeshInstance::set_navigation_mesh);
|
|
||||||
ClassDB::bind_method(D_METHOD("get_navigation_mesh"), &NavigationMeshInstance::get_navigation_mesh);
|
|
||||||
|
|
||||||
ClassDB::bind_method(D_METHOD("set_enabled", "enabled"), &NavigationMeshInstance::set_enabled);
|
ClassDB::bind_method(D_METHOD("set_enabled", "enabled"), &NavigationMeshInstance::set_enabled);
|
||||||
ClassDB::bind_method(D_METHOD("is_enabled"), &NavigationMeshInstance::is_enabled);
|
ClassDB::bind_method(D_METHOD("is_enabled"), &NavigationMeshInstance::is_enabled);
|
||||||
|
|
||||||
|
ClassDB::bind_method(D_METHOD("set_navigation_mesh", "navmesh"), &NavigationMeshInstance::set_navigation_mesh);
|
||||||
|
ClassDB::bind_method(D_METHOD("get_navigation_mesh"), &NavigationMeshInstance::get_navigation_mesh);
|
||||||
|
|
||||||
|
ClassDB::bind_method(D_METHOD("set_navigation_map", "navigation_map"), &NavigationMeshInstance::set_navigation_map);
|
||||||
|
ClassDB::bind_method(D_METHOD("get_navigation_map"), &NavigationMeshInstance::get_navigation_map);
|
||||||
|
|
||||||
ClassDB::bind_method(D_METHOD("set_navigation_layers", "navigation_layers"), &NavigationMeshInstance::set_navigation_layers);
|
ClassDB::bind_method(D_METHOD("set_navigation_layers", "navigation_layers"), &NavigationMeshInstance::set_navigation_layers);
|
||||||
ClassDB::bind_method(D_METHOD("get_navigation_layers"), &NavigationMeshInstance::get_navigation_layers);
|
ClassDB::bind_method(D_METHOD("get_navigation_layers"), &NavigationMeshInstance::get_navigation_layers);
|
||||||
|
|
||||||
ClassDB::bind_method(D_METHOD("get_region_rid"), &NavigationMeshInstance::get_region_rid);
|
|
||||||
|
|
||||||
ClassDB::bind_method(D_METHOD("set_enter_cost", "enter_cost"), &NavigationMeshInstance::set_enter_cost);
|
ClassDB::bind_method(D_METHOD("set_enter_cost", "enter_cost"), &NavigationMeshInstance::set_enter_cost);
|
||||||
ClassDB::bind_method(D_METHOD("get_enter_cost"), &NavigationMeshInstance::get_enter_cost);
|
ClassDB::bind_method(D_METHOD("get_enter_cost"), &NavigationMeshInstance::get_enter_cost);
|
||||||
|
|
||||||
ClassDB::bind_method(D_METHOD("set_travel_cost", "travel_cost"), &NavigationMeshInstance::set_travel_cost);
|
ClassDB::bind_method(D_METHOD("set_travel_cost", "travel_cost"), &NavigationMeshInstance::set_travel_cost);
|
||||||
ClassDB::bind_method(D_METHOD("get_travel_cost"), &NavigationMeshInstance::get_travel_cost);
|
ClassDB::bind_method(D_METHOD("get_travel_cost"), &NavigationMeshInstance::get_travel_cost);
|
||||||
|
|
||||||
|
ClassDB::bind_method(D_METHOD("get_region_rid"), &NavigationMeshInstance::get_region_rid);
|
||||||
|
|
||||||
ClassDB::bind_method(D_METHOD("bake_navigation_mesh", "on_thread"), &NavigationMeshInstance::bake_navigation_mesh, DEFVAL(true));
|
ClassDB::bind_method(D_METHOD("bake_navigation_mesh", "on_thread"), &NavigationMeshInstance::bake_navigation_mesh, DEFVAL(true));
|
||||||
ClassDB::bind_method(D_METHOD("_bake_finished", "nav_mesh"), &NavigationMeshInstance::_bake_finished);
|
ClassDB::bind_method(D_METHOD("_bake_finished"), &NavigationMeshInstance::_bake_finished);
|
||||||
ClassDB::bind_method(D_METHOD("_navigation_changed"), &NavigationMeshInstance::_navigation_changed);
|
ClassDB::bind_method(D_METHOD("_navigation_mesh_changed"), &NavigationMeshInstance::_navigation_mesh_changed);
|
||||||
|
|
||||||
#ifdef DEBUG_ENABLED
|
#ifdef DEBUG_ENABLED
|
||||||
ClassDB::bind_method(D_METHOD("_update_debug_mesh"), &NavigationMeshInstance::_update_debug_mesh);
|
ClassDB::bind_method(D_METHOD("_update_debug_mesh"), &NavigationMeshInstance::_update_debug_mesh);
|
||||||
@ -353,6 +357,8 @@ NavigationMeshInstance::NavigationMeshInstance() {
|
|||||||
|
|
||||||
navigation_layers = 1;
|
navigation_layers = 1;
|
||||||
|
|
||||||
|
baking_started = false;
|
||||||
|
|
||||||
region = NavigationServer::get_singleton()->region_create();
|
region = NavigationServer::get_singleton()->region_create();
|
||||||
NavigationServer::get_singleton()->region_set_enter_cost(region, get_enter_cost());
|
NavigationServer::get_singleton()->region_set_enter_cost(region, get_enter_cost());
|
||||||
NavigationServer::get_singleton()->region_set_travel_cost(region, get_travel_cost());
|
NavigationServer::get_singleton()->region_set_travel_cost(region, get_travel_cost());
|
||||||
|
@ -42,6 +42,7 @@ class NavigationMeshInstance : public Spatial {
|
|||||||
|
|
||||||
bool enabled;
|
bool enabled;
|
||||||
RID region;
|
RID region;
|
||||||
|
RID map_override;
|
||||||
Ref<NavigationMesh> navmesh;
|
Ref<NavigationMesh> navmesh;
|
||||||
|
|
||||||
real_t enter_cost = 0.0;
|
real_t enter_cost = 0.0;
|
||||||
@ -50,7 +51,8 @@ class NavigationMeshInstance : public Spatial {
|
|||||||
uint32_t navigation_layers = 1;
|
uint32_t navigation_layers = 1;
|
||||||
|
|
||||||
Navigation *navigation = nullptr;
|
Navigation *navigation = nullptr;
|
||||||
Thread bake_thread;
|
|
||||||
|
bool baking_started;
|
||||||
|
|
||||||
#ifdef DEBUG_ENABLED
|
#ifdef DEBUG_ENABLED
|
||||||
RID debug_instance;
|
RID debug_instance;
|
||||||
@ -76,8 +78,6 @@ public:
|
|||||||
void set_navigation_layers(uint32_t p_navigation_layers);
|
void set_navigation_layers(uint32_t p_navigation_layers);
|
||||||
uint32_t get_navigation_layers() const;
|
uint32_t get_navigation_layers() const;
|
||||||
|
|
||||||
RID get_region_rid() const;
|
|
||||||
|
|
||||||
void set_enter_cost(real_t p_enter_cost);
|
void set_enter_cost(real_t p_enter_cost);
|
||||||
real_t get_enter_cost() const;
|
real_t get_enter_cost() const;
|
||||||
|
|
||||||
@ -87,11 +87,16 @@ public:
|
|||||||
void set_navigation_mesh(const Ref<NavigationMesh> &p_navmesh);
|
void set_navigation_mesh(const Ref<NavigationMesh> &p_navmesh);
|
||||||
Ref<NavigationMesh> get_navigation_mesh() const;
|
Ref<NavigationMesh> get_navigation_mesh() const;
|
||||||
|
|
||||||
|
void set_navigation_map(RID p_navigation_map);
|
||||||
|
RID get_navigation_map() const;
|
||||||
|
|
||||||
|
RID get_region_rid() const;
|
||||||
|
|
||||||
/// Bakes the navigation mesh; once done, automatically
|
/// Bakes the navigation mesh; once done, automatically
|
||||||
/// sets the new navigation mesh and emits a signal
|
/// sets the new navigation mesh and emits a signal
|
||||||
void bake_navigation_mesh(bool p_on_thread);
|
void bake_navigation_mesh(bool p_on_thread = true);
|
||||||
void _bake_finished(Ref<NavigationMesh> p_nav_mesh);
|
void _bake_finished(Ref<NavigationMesh> p_nav_mesh);
|
||||||
void _navigation_changed();
|
void _navigation_mesh_changed();
|
||||||
|
|
||||||
String get_configuration_warning() const;
|
String get_configuration_warning() const;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user