Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
godotengine
GitHub Repository: godotengine/godot
Path: blob/master/modules/gltf/structures/gltf_node.cpp
10279 views
1
/**************************************************************************/
2
/* gltf_node.cpp */
3
/**************************************************************************/
4
/* This file is part of: */
5
/* GODOT ENGINE */
6
/* https://godotengine.org */
7
/**************************************************************************/
8
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
9
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
10
/* */
11
/* Permission is hereby granted, free of charge, to any person obtaining */
12
/* a copy of this software and associated documentation files (the */
13
/* "Software"), to deal in the Software without restriction, including */
14
/* without limitation the rights to use, copy, modify, merge, publish, */
15
/* distribute, sublicense, and/or sell copies of the Software, and to */
16
/* permit persons to whom the Software is furnished to do so, subject to */
17
/* the following conditions: */
18
/* */
19
/* The above copyright notice and this permission notice shall be */
20
/* included in all copies or substantial portions of the Software. */
21
/* */
22
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
23
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
24
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
25
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
26
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
27
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
28
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
29
/**************************************************************************/
30
31
#include "gltf_node.h"
32
33
#include "../gltf_state.h"
34
35
void GLTFNode::_bind_methods() {
36
ClassDB::bind_method(D_METHOD("get_original_name"), &GLTFNode::get_original_name);
37
ClassDB::bind_method(D_METHOD("set_original_name", "original_name"), &GLTFNode::set_original_name);
38
ClassDB::bind_method(D_METHOD("get_parent"), &GLTFNode::get_parent);
39
ClassDB::bind_method(D_METHOD("set_parent", "parent"), &GLTFNode::set_parent);
40
ClassDB::bind_method(D_METHOD("get_height"), &GLTFNode::get_height);
41
ClassDB::bind_method(D_METHOD("set_height", "height"), &GLTFNode::set_height);
42
ClassDB::bind_method(D_METHOD("get_xform"), &GLTFNode::get_xform);
43
ClassDB::bind_method(D_METHOD("set_xform", "xform"), &GLTFNode::set_xform);
44
ClassDB::bind_method(D_METHOD("get_mesh"), &GLTFNode::get_mesh);
45
ClassDB::bind_method(D_METHOD("set_mesh", "mesh"), &GLTFNode::set_mesh);
46
ClassDB::bind_method(D_METHOD("get_camera"), &GLTFNode::get_camera);
47
ClassDB::bind_method(D_METHOD("set_camera", "camera"), &GLTFNode::set_camera);
48
ClassDB::bind_method(D_METHOD("get_skin"), &GLTFNode::get_skin);
49
ClassDB::bind_method(D_METHOD("set_skin", "skin"), &GLTFNode::set_skin);
50
ClassDB::bind_method(D_METHOD("get_skeleton"), &GLTFNode::get_skeleton);
51
ClassDB::bind_method(D_METHOD("set_skeleton", "skeleton"), &GLTFNode::set_skeleton);
52
ClassDB::bind_method(D_METHOD("get_position"), &GLTFNode::get_position);
53
ClassDB::bind_method(D_METHOD("set_position", "position"), &GLTFNode::set_position);
54
ClassDB::bind_method(D_METHOD("get_rotation"), &GLTFNode::get_rotation);
55
ClassDB::bind_method(D_METHOD("set_rotation", "rotation"), &GLTFNode::set_rotation);
56
ClassDB::bind_method(D_METHOD("get_scale"), &GLTFNode::get_scale);
57
ClassDB::bind_method(D_METHOD("set_scale", "scale"), &GLTFNode::set_scale);
58
ClassDB::bind_method(D_METHOD("get_children"), &GLTFNode::get_children);
59
ClassDB::bind_method(D_METHOD("set_children", "children"), &GLTFNode::set_children);
60
ClassDB::bind_method(D_METHOD("append_child_index", "child_index"), &GLTFNode::append_child_index);
61
ClassDB::bind_method(D_METHOD("get_light"), &GLTFNode::get_light);
62
ClassDB::bind_method(D_METHOD("set_light", "light"), &GLTFNode::set_light);
63
ClassDB::bind_method(D_METHOD("get_visible"), &GLTFNode::get_visible);
64
ClassDB::bind_method(D_METHOD("set_visible", "visible"), &GLTFNode::set_visible);
65
ClassDB::bind_method(D_METHOD("get_additional_data", "extension_name"), &GLTFNode::get_additional_data);
66
ClassDB::bind_method(D_METHOD("set_additional_data", "extension_name", "additional_data"), &GLTFNode::set_additional_data);
67
ClassDB::bind_method(D_METHOD("get_scene_node_path", "gltf_state", "handle_skeletons"), &GLTFNode::get_scene_node_path, DEFVAL(true));
68
69
ADD_PROPERTY(PropertyInfo(Variant::STRING, "original_name"), "set_original_name", "get_original_name"); // String
70
ADD_PROPERTY(PropertyInfo(Variant::INT, "parent"), "set_parent", "get_parent"); // GLTFNodeIndex
71
ADD_PROPERTY(PropertyInfo(Variant::INT, "height"), "set_height", "get_height"); // int
72
ADD_PROPERTY(PropertyInfo(Variant::TRANSFORM3D, "xform"), "set_xform", "get_xform"); // Transform3D
73
ADD_PROPERTY(PropertyInfo(Variant::INT, "mesh"), "set_mesh", "get_mesh"); // GLTFMeshIndex
74
ADD_PROPERTY(PropertyInfo(Variant::INT, "camera"), "set_camera", "get_camera"); // GLTFCameraIndex
75
ADD_PROPERTY(PropertyInfo(Variant::INT, "skin"), "set_skin", "get_skin"); // GLTFSkinIndex
76
ADD_PROPERTY(PropertyInfo(Variant::INT, "skeleton"), "set_skeleton", "get_skeleton"); // GLTFSkeletonIndex
77
ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "position"), "set_position", "get_position"); // Vector3
78
ADD_PROPERTY(PropertyInfo(Variant::QUATERNION, "rotation"), "set_rotation", "get_rotation"); // Quaternion
79
ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "scale"), "set_scale", "get_scale"); // Vector3
80
ADD_PROPERTY(PropertyInfo(Variant::PACKED_INT32_ARRAY, "children"), "set_children", "get_children"); // Vector<int>
81
ADD_PROPERTY(PropertyInfo(Variant::INT, "light"), "set_light", "get_light"); // GLTFLightIndex
82
ADD_PROPERTY(PropertyInfo(Variant::BOOL, "visible"), "set_visible", "get_visible"); // bool
83
}
84
85
String GLTFNode::get_original_name() {
86
return original_name;
87
}
88
void GLTFNode::set_original_name(String p_name) {
89
original_name = p_name;
90
}
91
92
GLTFNodeIndex GLTFNode::get_parent() {
93
return parent;
94
}
95
96
void GLTFNode::set_parent(GLTFNodeIndex p_parent) {
97
parent = p_parent;
98
}
99
100
int GLTFNode::get_height() {
101
return height;
102
}
103
104
void GLTFNode::set_height(int p_height) {
105
height = p_height;
106
}
107
108
Transform3D GLTFNode::get_xform() {
109
return transform;
110
}
111
112
void GLTFNode::set_xform(Transform3D p_xform) {
113
transform = p_xform;
114
}
115
116
GLTFMeshIndex GLTFNode::get_mesh() {
117
return mesh;
118
}
119
120
void GLTFNode::set_mesh(GLTFMeshIndex p_mesh) {
121
mesh = p_mesh;
122
}
123
124
GLTFCameraIndex GLTFNode::get_camera() {
125
return camera;
126
}
127
128
void GLTFNode::set_camera(GLTFCameraIndex p_camera) {
129
camera = p_camera;
130
}
131
132
GLTFSkinIndex GLTFNode::get_skin() {
133
return skin;
134
}
135
136
void GLTFNode::set_skin(GLTFSkinIndex p_skin) {
137
skin = p_skin;
138
}
139
140
GLTFSkeletonIndex GLTFNode::get_skeleton() {
141
return skeleton;
142
}
143
144
void GLTFNode::set_skeleton(GLTFSkeletonIndex p_skeleton) {
145
skeleton = p_skeleton;
146
}
147
148
Vector3 GLTFNode::get_position() {
149
return transform.origin;
150
}
151
152
void GLTFNode::set_position(Vector3 p_position) {
153
transform.origin = p_position;
154
}
155
156
Quaternion GLTFNode::get_rotation() {
157
return transform.basis.get_rotation_quaternion();
158
}
159
160
void GLTFNode::set_rotation(Quaternion p_rotation) {
161
transform.basis.set_quaternion_scale(p_rotation, transform.basis.get_scale());
162
}
163
164
Vector3 GLTFNode::get_scale() {
165
return transform.basis.get_scale();
166
}
167
168
void GLTFNode::set_scale(Vector3 p_scale) {
169
transform.basis = transform.basis.orthonormalized() * Basis::from_scale(p_scale);
170
}
171
172
Vector<int> GLTFNode::get_children() {
173
return children;
174
}
175
176
void GLTFNode::set_children(Vector<int> p_children) {
177
children = p_children;
178
}
179
180
void GLTFNode::append_child_index(int p_child_index) {
181
children.append(p_child_index);
182
}
183
184
GLTFLightIndex GLTFNode::get_light() {
185
return light;
186
}
187
188
void GLTFNode::set_light(GLTFLightIndex p_light) {
189
light = p_light;
190
}
191
192
bool GLTFNode::get_visible() {
193
return visible;
194
}
195
196
void GLTFNode::set_visible(bool p_visible) {
197
visible = p_visible;
198
}
199
200
Variant GLTFNode::get_additional_data(const StringName &p_extension_name) {
201
return additional_data[p_extension_name];
202
}
203
204
bool GLTFNode::has_additional_data(const StringName &p_extension_name) {
205
return additional_data.has(p_extension_name);
206
}
207
208
void GLTFNode::set_additional_data(const StringName &p_extension_name, Variant p_additional_data) {
209
additional_data[p_extension_name] = p_additional_data;
210
}
211
212
NodePath GLTFNode::get_scene_node_path(Ref<GLTFState> p_state, bool p_handle_skeletons) {
213
Vector<StringName> path;
214
Vector<StringName> subpath;
215
Ref<GLTFNode> current_gltf_node = this;
216
const int gltf_node_count = p_state->nodes.size();
217
if (p_handle_skeletons && skeleton != -1) {
218
// Special case for skeleton nodes, skip all bones so that the path is to the Skeleton3D node.
219
// A path that would otherwise be `A/B/C/Bone1/Bone2/Bone3` becomes `A/B/C/Skeleton3D:Bone3`.
220
subpath.append(get_name());
221
// The generated Skeleton3D node will be named Skeleton3D, so add it to the path.
222
path.append("Skeleton3D");
223
do {
224
const int parent_index = current_gltf_node->get_parent();
225
ERR_FAIL_INDEX_V(parent_index, gltf_node_count, NodePath());
226
current_gltf_node = p_state->nodes[parent_index];
227
} while (current_gltf_node->skeleton != -1);
228
}
229
const bool is_godot_single_root = p_state->extensions_used.has("GODOT_single_root");
230
while (true) {
231
const int parent_index = current_gltf_node->get_parent();
232
if (is_godot_single_root && parent_index == -1) {
233
// For GODOT_single_root scenes, the root glTF node becomes the Godot scene root, so it
234
// should not be included in the path. Ex: A/B/C, A is single root, we want B/C only.
235
break;
236
}
237
path.insert(0, current_gltf_node->get_name());
238
if (!is_godot_single_root && parent_index == -1) {
239
break;
240
}
241
ERR_FAIL_INDEX_V(parent_index, gltf_node_count, NodePath());
242
current_gltf_node = p_state->nodes[parent_index];
243
}
244
if (unlikely(path.is_empty())) {
245
path.append(".");
246
}
247
return NodePath(path, subpath, false);
248
}
249
250