Make Mesh::generate_triangle_mesh()
handle PRIMITIVE_TRIANGLE_STRIP
This commit is contained in:
parent
532e253a7c
commit
9b556c96f0
2 changed files with 47 additions and 18 deletions
|
@ -123,7 +123,7 @@
|
||||||
<method name="generate_triangle_mesh" qualifiers="const">
|
<method name="generate_triangle_mesh" qualifiers="const">
|
||||||
<return type="TriangleMesh" />
|
<return type="TriangleMesh" />
|
||||||
<description>
|
<description>
|
||||||
Generate a [TriangleMesh] from the mesh.
|
Generate a [TriangleMesh] from the mesh. Considers only surfaces using one of these primitive types: [constant PRIMITIVE_TRIANGLES], [constant PRIMITIVE_TRIANGLE_STRIP].
|
||||||
</description>
|
</description>
|
||||||
</method>
|
</method>
|
||||||
<method name="get_aabb" qualifiers="const">
|
<method name="get_aabb" qualifiers="const">
|
||||||
|
|
|
@ -161,32 +161,45 @@ Ref<TriangleMesh> Mesh::generate_triangle_mesh() const {
|
||||||
return triangle_mesh;
|
return triangle_mesh;
|
||||||
}
|
}
|
||||||
|
|
||||||
int facecount = 0;
|
int faces_size = 0;
|
||||||
|
|
||||||
for (int i = 0; i < get_surface_count(); i++) {
|
for (int i = 0; i < get_surface_count(); i++) {
|
||||||
if (surface_get_primitive_type(i) != PRIMITIVE_TRIANGLES) {
|
switch (surface_get_primitive_type(i)) {
|
||||||
continue;
|
case PRIMITIVE_TRIANGLES: {
|
||||||
}
|
int len = (surface_get_format(i) & ARRAY_FORMAT_INDEX) ? surface_get_array_index_len(i) : surface_get_array_len(i);
|
||||||
|
// Don't error if zero, it's valid (we'll just skip it later).
|
||||||
if (surface_get_format(i) & ARRAY_FORMAT_INDEX) {
|
ERR_CONTINUE_MSG((len % 3) != 0, vformat("Ignoring surface %d, incorrect %s count: %d (for PRIMITIVE_TRIANGLES).", i, (surface_get_format(i) & ARRAY_FORMAT_INDEX) ? "index" : "vertex", len));
|
||||||
facecount += surface_get_array_index_len(i);
|
faces_size += len;
|
||||||
} else {
|
} break;
|
||||||
facecount += surface_get_array_len(i);
|
case PRIMITIVE_TRIANGLE_STRIP: {
|
||||||
|
int len = (surface_get_format(i) & ARRAY_FORMAT_INDEX) ? surface_get_array_index_len(i) : surface_get_array_len(i);
|
||||||
|
// Don't error if zero, it's valid (we'll just skip it later).
|
||||||
|
ERR_CONTINUE_MSG(len != 0 && len < 3, vformat("Ignoring surface %d, incorrect %s count: %d (for PRIMITIVE_TRIANGLE_STRIP).", i, (surface_get_format(i) & ARRAY_FORMAT_INDEX) ? "index" : "vertex", len));
|
||||||
|
faces_size += (len == 0) ? 0 : (len - 2) * 3;
|
||||||
|
} break;
|
||||||
|
default: {
|
||||||
|
} break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (facecount == 0 || (facecount % 3) != 0) {
|
if (faces_size == 0) {
|
||||||
return triangle_mesh;
|
return triangle_mesh;
|
||||||
}
|
}
|
||||||
|
|
||||||
Vector<Vector3> faces;
|
Vector<Vector3> faces;
|
||||||
faces.resize(facecount);
|
faces.resize(faces_size);
|
||||||
Vector3 *facesw = faces.ptrw();
|
Vector3 *facesw = faces.ptrw();
|
||||||
|
|
||||||
int widx = 0;
|
int widx = 0;
|
||||||
|
|
||||||
for (int i = 0; i < get_surface_count(); i++) {
|
for (int i = 0; i < get_surface_count(); i++) {
|
||||||
if (surface_get_primitive_type(i) != PRIMITIVE_TRIANGLES) {
|
Mesh::PrimitiveType primitive = surface_get_primitive_type(i);
|
||||||
|
if (primitive != PRIMITIVE_TRIANGLES && primitive != PRIMITIVE_TRIANGLE_STRIP) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
int len = (surface_get_format(i) & ARRAY_FORMAT_INDEX) ? surface_get_array_index_len(i) : surface_get_array_len(i);
|
||||||
|
if ((primitive == PRIMITIVE_TRIANGLES && (len == 0 || (len % 3) != 0)) || (primitive == PRIMITIVE_TRIANGLE_STRIP && len < 3)) {
|
||||||
|
// Error was already shown, just skip (including zero).
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -202,15 +215,31 @@ Ref<TriangleMesh> Mesh::generate_triangle_mesh() const {
|
||||||
Vector<int> indices = a[ARRAY_INDEX];
|
Vector<int> indices = a[ARRAY_INDEX];
|
||||||
const int *ir = indices.ptr();
|
const int *ir = indices.ptr();
|
||||||
|
|
||||||
|
if (primitive == PRIMITIVE_TRIANGLES) {
|
||||||
for (int j = 0; j < ic; j++) {
|
for (int j = 0; j < ic; j++) {
|
||||||
int index = ir[j];
|
int index = ir[j];
|
||||||
facesw[widx++] = vr[index];
|
facesw[widx++] = vr[index];
|
||||||
}
|
}
|
||||||
|
} else { // PRIMITIVE_TRIANGLE_STRIP
|
||||||
|
for (int j = 2; j < ic; j++) {
|
||||||
|
facesw[widx++] = vr[ir[j - 2]];
|
||||||
|
facesw[widx++] = vr[ir[j - 1]];
|
||||||
|
facesw[widx++] = vr[ir[j]];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
if (primitive == PRIMITIVE_TRIANGLES) {
|
||||||
for (int j = 0; j < vc; j++) {
|
for (int j = 0; j < vc; j++) {
|
||||||
facesw[widx++] = vr[j];
|
facesw[widx++] = vr[j];
|
||||||
}
|
}
|
||||||
|
} else { // PRIMITIVE_TRIANGLE_STRIP
|
||||||
|
for (int j = 2; j < vc; j++) {
|
||||||
|
facesw[widx++] = vr[j - 2];
|
||||||
|
facesw[widx++] = vr[j - 1];
|
||||||
|
facesw[widx++] = vr[j];
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue