godot/editor/spatial_editor_gizmos.h

512 lines
16 KiB
C++
Raw Normal View History

2015-10-09 02:00:40 +08:00
/*************************************************************************/
/* spatial_editor_gizmos.h */
/*************************************************************************/
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
2015-10-09 02:00:40 +08:00
/*************************************************************************/
/* Copyright (c) 2007-2018 Juan Linietsky, Ariel Manzur. */
/* Copyright (c) 2014-2018 Godot Engine contributors (cf. AUTHORS.md) */
2015-10-09 02:00:40 +08:00
/* */
/* 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. */
/*************************************************************************/
2015-10-09 02:00:40 +08:00
#ifndef SPATIAL_EDITOR_GIZMOS_H
#define SPATIAL_EDITOR_GIZMOS_H
#include "editor/plugins/spatial_editor_plugin.h"
#include "scene/3d/audio_stream_player_3d.h"
#include "scene/3d/baked_lightmap.h"
#include "scene/3d/camera.h"
2017-04-09 09:38:11 +08:00
#include "scene/3d/collision_polygon.h"
#include "scene/3d/collision_shape.h"
#include "scene/3d/gi_probe.h"
2015-10-09 02:00:40 +08:00
#include "scene/3d/light.h"
2016-03-20 10:10:04 +08:00
#include "scene/3d/listener.h"
2015-10-09 02:00:40 +08:00
#include "scene/3d/mesh_instance.h"
#include "scene/3d/navigation_mesh.h"
2017-04-09 09:38:11 +08:00
#include "scene/3d/particles.h"
#include "scene/3d/physics_joint.h"
2015-10-09 02:00:40 +08:00
#include "scene/3d/portal.h"
#include "scene/3d/position_3d.h"
2015-10-09 02:00:40 +08:00
#include "scene/3d/ray_cast.h"
2016-11-20 00:23:37 +08:00
#include "scene/3d/reflection_probe.h"
#include "scene/3d/room_instance.h"
#include "scene/3d/vehicle_body.h"
2017-04-09 09:38:11 +08:00
#include "scene/3d/visibility_notifier.h"
2015-10-09 02:00:40 +08:00
class Camera;
class EditorSpatialGizmo : public SpatialEditorGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(EditorSpatialGizmo, SpatialGizmo);
2015-10-09 02:00:40 +08:00
struct Instance {
2015-10-09 02:00:40 +08:00
RID instance;
Ref<ArrayMesh> mesh;
2015-10-09 02:00:40 +08:00
RID skeleton;
bool billboard;
bool unscaled;
bool can_intersect;
bool extra_margin;
Instance() {
billboard = false;
unscaled = false;
can_intersect = false;
extra_margin = false;
2015-10-09 02:00:40 +08:00
}
void create_instance(Spatial *p_base);
};
Vector<Vector3> collision_segments;
Ref<TriangleMesh> collision_mesh;
2017-11-17 10:09:00 +08:00
AABB collision_mesh_bounds;
2015-10-09 02:00:40 +08:00
struct Handle {
Vector3 pos;
bool billboard;
};
Vector<Vector3> handles;
Vector<Vector3> secondary_handles;
bool billboard_handle;
bool valid;
Spatial *base;
Vector<Instance> instances;
Spatial *spatial_node;
void _set_spatial_node(Node *p_node) { set_spatial_node(Object::cast_to<Spatial>(p_node)); }
2015-10-09 02:00:40 +08:00
protected:
void add_lines(const Vector<Vector3> &p_lines, const Ref<Material> &p_material, bool p_billboard = false);
void add_mesh(const Ref<ArrayMesh> &p_mesh, bool p_billboard = false, const RID &p_skeleton = RID());
2015-10-09 02:00:40 +08:00
void add_collision_segments(const Vector<Vector3> &p_lines);
2017-11-17 10:09:00 +08:00
void add_collision_triangles(const Ref<TriangleMesh> &p_tmesh, const AABB &p_bounds = AABB());
void add_unscaled_billboard(const Ref<Material> &p_material, float p_scale = 1);
void add_handles(const Vector<Vector3> &p_handles, bool p_billboard = false, bool p_secondary = false);
void add_solid_box(Ref<Material> &p_material, Vector3 p_size, Vector3 p_position = Vector3());
2015-10-09 02:00:40 +08:00
void set_spatial_node(Spatial *p_node);
2017-12-16 21:05:19 +08:00
const Spatial *get_spatial_node() const { return spatial_node; }
2015-10-09 02:00:40 +08:00
static void _bind_methods();
2015-10-09 02:00:40 +08:00
Ref<SpatialMaterial> create_material(const String &p_name, const Color &p_color, bool p_billboard = false, bool p_on_top = false, bool p_use_vertex_color = false);
Ref<SpatialMaterial> create_icon_material(const String &p_name, const Ref<Texture> &p_texture, bool p_on_top = false, const Color &p_albedo = Color(1, 1, 1, 1));
public:
2015-10-09 02:00:40 +08:00
virtual Vector3 get_handle_pos(int p_idx) const;
virtual bool intersect_frustum(const Camera *p_camera, const Vector<Plane> &p_frustum);
virtual bool intersect_ray(const Camera *p_camera, const Point2 &p_point, Vector3 &r_pos, Vector3 &r_normal, int *r_gizmo_handle = NULL, bool p_sec_first = false);
2015-10-09 02:00:40 +08:00
void clear();
void create();
void transform();
virtual void redraw();
2015-10-09 02:00:40 +08:00
void free();
virtual bool is_editable() const;
virtual bool can_draw() const;
2015-10-09 02:00:40 +08:00
EditorSpatialGizmo();
~EditorSpatialGizmo();
2015-10-09 02:00:40 +08:00
};
class LightSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(LightSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
Light *light;
2015-10-09 02:00:40 +08:00
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
2015-10-09 02:00:40 +08:00
void redraw();
LightSpatialGizmo(Light *p_light = NULL);
2015-10-09 02:00:40 +08:00
};
class AudioStreamPlayer3DSpatialGizmo : public EditorSpatialGizmo {
GDCLASS(AudioStreamPlayer3DSpatialGizmo, EditorSpatialGizmo);
AudioStreamPlayer3D *player;
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
void redraw();
AudioStreamPlayer3DSpatialGizmo(AudioStreamPlayer3D *p_player = NULL);
};
class CameraSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(CameraSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
Camera *camera;
2015-10-09 02:00:40 +08:00
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
2015-10-09 02:00:40 +08:00
void redraw();
CameraSpatialGizmo(Camera *p_camera = NULL);
2015-10-09 02:00:40 +08:00
};
class MeshInstanceSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(MeshInstanceSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
MeshInstance *mesh;
2015-10-09 02:00:40 +08:00
public:
virtual bool can_draw() const;
2015-10-09 02:00:40 +08:00
void redraw();
MeshInstanceSpatialGizmo(MeshInstance *p_mesh = NULL);
2015-10-09 02:00:40 +08:00
};
class Position3DSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(Position3DSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
Position3D *p3d;
2015-10-09 02:00:40 +08:00
public:
void redraw();
Position3DSpatialGizmo(Position3D *p_p3d = NULL);
2015-10-09 02:00:40 +08:00
};
class SkeletonSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(SkeletonSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
Skeleton *skel;
2015-10-09 02:00:40 +08:00
public:
void redraw();
SkeletonSpatialGizmo(Skeleton *p_skel = NULL);
2015-10-09 02:00:40 +08:00
};
class PhysicalBoneSpatialGizmo : public EditorSpatialGizmo {
GDCLASS(PhysicalBoneSpatialGizmo, EditorSpatialGizmo);
PhysicalBone *physical_bone;
public:
//virtual Transform get_global_gizmo_transform();
virtual void redraw();
PhysicalBoneSpatialGizmo(PhysicalBone *p_pb = NULL);
};
#if 0
class PortalSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(PortalSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
Portal *portal;
2015-10-09 02:00:40 +08:00
public:
void redraw();
PortalSpatialGizmo(Portal *p_portal = NULL);
2015-10-09 02:00:40 +08:00
};
#endif
2015-10-09 02:00:40 +08:00
class VisibilityNotifierGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(VisibilityNotifierGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
VisibilityNotifier *notifier;
2015-10-09 02:00:40 +08:00
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
2015-10-09 02:00:40 +08:00
void redraw();
VisibilityNotifierGizmo(VisibilityNotifier *p_notifier = NULL);
2015-10-09 02:00:40 +08:00
};
2017-04-09 09:38:11 +08:00
class ParticlesGizmo : public EditorSpatialGizmo {
GDCLASS(ParticlesGizmo, EditorSpatialGizmo);
Particles *particles;
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
void redraw();
ParticlesGizmo(Particles *p_particles = NULL);
};
class ReflectionProbeGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(ReflectionProbeGizmo, EditorSpatialGizmo);
2016-11-20 00:23:37 +08:00
ReflectionProbe *probe;
2016-11-20 00:23:37 +08:00
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
2016-11-20 00:23:37 +08:00
void redraw();
2017-08-13 00:52:50 +08:00
ReflectionProbeGizmo(ReflectionProbe *p_probe = NULL);
2016-11-20 00:23:37 +08:00
};
class GIProbeGizmo : public EditorSpatialGizmo {
2016-12-20 11:21:07 +08:00
GDCLASS(GIProbeGizmo, EditorSpatialGizmo);
2016-12-20 11:21:07 +08:00
GIProbe *probe;
2016-12-20 11:21:07 +08:00
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
2016-12-20 11:21:07 +08:00
void redraw();
2017-08-13 00:52:50 +08:00
GIProbeGizmo(GIProbe *p_probe = NULL);
2016-12-20 11:21:07 +08:00
};
class BakedIndirectLightGizmo : public EditorSpatialGizmo {
GDCLASS(BakedIndirectLightGizmo, EditorSpatialGizmo);
BakedLightmap *baker;
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
void redraw();
BakedIndirectLightGizmo(BakedLightmap *p_baker = NULL);
};
class CollisionShapeSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(CollisionShapeSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
CollisionShape *cs;
2015-10-09 02:00:40 +08:00
public:
virtual String get_handle_name(int p_idx) const;
virtual Variant get_handle_value(int p_idx) const;
virtual void set_handle(int p_idx, Camera *p_camera, const Point2 &p_point);
virtual void commit_handle(int p_idx, const Variant &p_restore, bool p_cancel = false);
2015-10-09 02:00:40 +08:00
void redraw();
CollisionShapeSpatialGizmo(CollisionShape *p_cs = NULL);
2015-10-09 02:00:40 +08:00
};
class CollisionPolygonSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(CollisionPolygonSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
CollisionPolygon *polygon;
2015-10-09 02:00:40 +08:00
public:
void redraw();
CollisionPolygonSpatialGizmo(CollisionPolygon *p_polygon = NULL);
2015-10-09 02:00:40 +08:00
};
class RayCastSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(RayCastSpatialGizmo, EditorSpatialGizmo);
2016-11-20 00:23:37 +08:00
RayCast *raycast;
2015-10-09 02:00:40 +08:00
public:
void redraw();
RayCastSpatialGizmo(RayCast *p_raycast = NULL);
2015-10-09 02:00:40 +08:00
};
class VehicleWheelSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(VehicleWheelSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
VehicleWheel *car_wheel;
2015-10-09 02:00:40 +08:00
public:
void redraw();
VehicleWheelSpatialGizmo(VehicleWheel *p_car_wheel = NULL);
2015-10-09 02:00:40 +08:00
};
class NavigationMeshSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(NavigationMeshSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
struct _EdgeKey {
Vector3 from;
Vector3 to;
bool operator<(const _EdgeKey &p_with) const { return from == p_with.from ? to < p_with.to : from < p_with.from; }
2015-10-09 02:00:40 +08:00
};
NavigationMeshInstance *navmesh;
2015-10-09 02:00:40 +08:00
public:
void redraw();
NavigationMeshSpatialGizmo(NavigationMeshInstance *p_navmesh = NULL);
2015-10-09 02:00:40 +08:00
};
2017-12-16 21:05:19 +08:00
class JointGizmosDrawer {
public:
static Basis look_body(const Transform &p_joint_transform, const Transform &p_body_transform);
2017-12-16 21:05:19 +08:00
static Basis look_body_toward(Vector3::Axis p_axis, const Transform &joint_transform, const Transform &body_transform);
static Basis look_body_toward_x(const Transform &p_joint_transform, const Transform &p_body_transform);
static Basis look_body_toward_y(const Transform &p_joint_transform, const Transform &p_body_transform);
2017-12-16 21:05:19 +08:00
/// Special function just used for physics joints, it that returns a basis constrained toward Joint Z axis
/// with axis X and Y that are looking toward the body and oriented toward up
static Basis look_body_toward_z(const Transform &p_joint_transform, const Transform &p_body_transform);
2017-12-16 21:05:19 +08:00
// Draw circle around p_axis
static void draw_circle(Vector3::Axis p_axis, real_t p_radius, const Transform &p_offset, const Basis &p_base, real_t p_limit_lower, real_t p_limit_upper, Vector<Vector3> &r_points, bool p_inverse = false);
static void draw_cone(const Transform &p_offset, const Basis &p_base, real_t p_swing, real_t p_twist, Vector<Vector3> &r_points);
};
class PinJointSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(PinJointSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
PinJoint *p3d;
2015-10-09 02:00:40 +08:00
public:
2017-12-16 21:05:19 +08:00
static void CreateGizmo(const Transform &p_offset, Vector<Vector3> &r_cursor_points);
2015-10-09 02:00:40 +08:00
void redraw();
PinJointSpatialGizmo(PinJoint *p_p3d = NULL);
2015-10-09 02:00:40 +08:00
};
class HingeJointSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(HingeJointSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
HingeJoint *p3d;
2015-10-09 02:00:40 +08:00
public:
2017-12-16 21:05:19 +08:00
static void CreateGizmo(const Transform &p_offset, const Transform &p_trs_joint, const Transform &p_trs_body_a, const Transform &p_trs_body_b, real_t p_limit_lower, real_t p_limit_upper, bool p_use_limit, Vector<Vector3> &r_common_points, Vector<Vector3> *r_body_a_points, Vector<Vector3> *r_body_b_points);
2015-10-09 02:00:40 +08:00
void redraw();
HingeJointSpatialGizmo(HingeJoint *p_p3d = NULL);
2015-10-09 02:00:40 +08:00
};
class SliderJointSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(SliderJointSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
SliderJoint *p3d;
2015-10-09 02:00:40 +08:00
public:
2017-12-16 21:05:19 +08:00
static void CreateGizmo(const Transform &p_offset, const Transform &p_trs_joint, const Transform &p_trs_body_a, const Transform &p_trs_body_b, real_t p_angular_limit_lower, real_t p_angular_limit_upper, real_t p_linear_limit_lower, real_t p_linear_limit_upper, Vector<Vector3> &r_points, Vector<Vector3> *r_body_a_points, Vector<Vector3> *r_body_b_points);
2015-10-09 02:00:40 +08:00
void redraw();
SliderJointSpatialGizmo(SliderJoint *p_p3d = NULL);
2015-10-09 02:00:40 +08:00
};
class ConeTwistJointSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(ConeTwistJointSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
ConeTwistJoint *p3d;
2015-10-09 02:00:40 +08:00
public:
2017-12-16 21:05:19 +08:00
static void CreateGizmo(const Transform &p_offset, const Transform &p_trs_joint, const Transform &p_trs_body_a, const Transform &p_trs_body_b, real_t p_swing, real_t p_twist, Vector<Vector3> &r_points, Vector<Vector3> *r_body_a_points, Vector<Vector3> *r_body_b_points);
2015-10-09 02:00:40 +08:00
void redraw();
ConeTwistJointSpatialGizmo(ConeTwistJoint *p_p3d = NULL);
2015-10-09 02:00:40 +08:00
};
class Generic6DOFJointSpatialGizmo : public EditorSpatialGizmo {
2015-10-09 02:00:40 +08:00
GDCLASS(Generic6DOFJointSpatialGizmo, EditorSpatialGizmo);
2015-10-09 02:00:40 +08:00
Generic6DOFJoint *p3d;
2015-10-09 02:00:40 +08:00
public:
2017-12-16 21:05:19 +08:00
static void CreateGizmo(
const Transform &p_offset,
const Transform &p_trs_joint,
const Transform &p_trs_body_a,
const Transform &p_trs_body_b,
real_t p_angular_limit_lower_x,
real_t p_angular_limit_upper_x,
real_t p_linear_limit_lower_x,
real_t p_linear_limit_upper_x,
bool p_enable_angular_limit_x,
bool p_enable_linear_limit_x,
real_t p_angular_limit_lower_y,
real_t p_angular_limit_upper_y,
real_t p_linear_limit_lower_y,
real_t p_linear_limit_upper_y,
bool p_enable_angular_limit_y,
bool p_enable_linear_limit_y,
real_t p_angular_limit_lower_z,
real_t p_angular_limit_upper_z,
real_t p_linear_limit_lower_z,
real_t p_linear_limit_upper_z,
bool p_enable_angular_limit_z,
bool p_enable_linear_limit_z,
Vector<Vector3> &r_points,
Vector<Vector3> *r_body_a_points,
Vector<Vector3> *r_body_b_points);
2015-10-09 02:00:40 +08:00
void redraw();
Generic6DOFJointSpatialGizmo(Generic6DOFJoint *p_p3d = NULL);
2015-10-09 02:00:40 +08:00
};
class SpatialEditorGizmos {
2015-10-09 02:00:40 +08:00
public:
HashMap<String, Ref<SpatialMaterial> > material_cache;
Ref<SpatialMaterial> handle2_material;
Ref<SpatialMaterial> handle2_material_billboard;
Ref<SpatialMaterial> handle_material;
Ref<SpatialMaterial> handle_material_billboard;
2015-10-09 02:00:40 +08:00
Ref<Texture> handle_t;
Ref<ArrayMesh> pos3d_mesh;
Ref<ArrayMesh> listener_line_mesh;
2015-10-09 02:00:40 +08:00
static SpatialEditorGizmos *singleton;
Ref<SpatialEditorGizmo> get_gizmo(Spatial *p_spatial);
SpatialEditorGizmos();
};
#endif // SPATIAL_EDITOR_GIZMOS_H