mirror of
https://github.com/Relintai/voxelman.git
synced 2025-01-17 14:47:21 +01:00
138 lines
2.9 KiB
C
138 lines
2.9 KiB
C
|
#ifndef VOXEL_MESHER_DMC_H
|
||
|
#define VOXEL_MESHER_DMC_H
|
||
|
|
||
|
#include "../../util/object_pool.h"
|
||
|
#include "../voxel_mesher.h"
|
||
|
#include "hermite_value.h"
|
||
|
#include "mesh_builder.h"
|
||
|
#include <scene/resources/mesh.h>
|
||
|
|
||
|
namespace dmc {
|
||
|
|
||
|
struct OctreeNode;
|
||
|
typedef ObjectPool<OctreeNode> OctreeNodePool;
|
||
|
|
||
|
// Octree used only for dual grid construction
|
||
|
struct OctreeNode {
|
||
|
|
||
|
Vector3i origin;
|
||
|
int size; // Nodes are cubic
|
||
|
HermiteValue center_value;
|
||
|
OctreeNode *children[8];
|
||
|
|
||
|
OctreeNode() {
|
||
|
init();
|
||
|
}
|
||
|
|
||
|
inline void init() {
|
||
|
for (int i = 0; i < 8; ++i) {
|
||
|
children[i] = nullptr;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void recycle(OctreeNodePool &pool) {
|
||
|
for (int i = 0; i < 8; ++i) {
|
||
|
if (children[i]) {
|
||
|
pool.recycle(children[i]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
inline bool has_children() const {
|
||
|
return children[0] != nullptr;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
struct DualCell {
|
||
|
Vector3 corners[8];
|
||
|
HermiteValue values[8];
|
||
|
bool has_values = false;
|
||
|
|
||
|
inline void set_corner(int i, Vector3 vertex, HermiteValue value) {
|
||
|
CRASH_COND(i < 0 || i >= 8);
|
||
|
corners[i] = vertex;
|
||
|
values[i] = value;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
struct DualGrid {
|
||
|
std::vector<DualCell> cells;
|
||
|
};
|
||
|
|
||
|
} // namespace dmc
|
||
|
|
||
|
// Mesher extending Marching Cubes using a dual grid.
|
||
|
class VoxelMesherDMC : public VoxelMesher {
|
||
|
GDCLASS(VoxelMesherDMC, VoxelMesher)
|
||
|
public:
|
||
|
static const int MINIMUM_PADDING = 2;
|
||
|
|
||
|
enum MeshMode {
|
||
|
MESH_NORMAL,
|
||
|
MESH_WIREFRAME,
|
||
|
MESH_DEBUG_OCTREE,
|
||
|
MESH_DEBUG_DUAL_GRID
|
||
|
};
|
||
|
|
||
|
enum SimplifyMode {
|
||
|
SIMPLIFY_OCTREE_BOTTOM_UP,
|
||
|
SIMPLIFY_OCTREE_TOP_DOWN,
|
||
|
SIMPLIFY_NONE
|
||
|
};
|
||
|
|
||
|
enum SeamMode {
|
||
|
SEAM_NONE, // No seam management
|
||
|
SEAM_MARCHING_SQUARE_SKIRTS,
|
||
|
// SEAM_OVERLAP // Polygonize extra voxels with lower isolevel
|
||
|
// SEAM_JUNCTION_TRIANGLES // Extra triangles for alternate borders, requires index buffer switching
|
||
|
};
|
||
|
|
||
|
VoxelMesherDMC();
|
||
|
|
||
|
void set_mesh_mode(MeshMode mode);
|
||
|
MeshMode get_mesh_mode() const;
|
||
|
|
||
|
void set_simplify_mode(SimplifyMode mode);
|
||
|
SimplifyMode get_simplify_mode() const;
|
||
|
|
||
|
void set_geometric_error(real_t geometric_error);
|
||
|
float get_geometric_error() const;
|
||
|
|
||
|
void set_seam_mode(SeamMode mode);
|
||
|
SeamMode get_seam_mode() const;
|
||
|
|
||
|
void build(VoxelMesher::Output &output, const VoxelBuffer &voxels, int padding) override;
|
||
|
int get_minimum_padding() const override;
|
||
|
|
||
|
Dictionary get_stats() const;
|
||
|
|
||
|
VoxelMesher *clone() override;
|
||
|
|
||
|
protected:
|
||
|
static void _bind_methods();
|
||
|
|
||
|
private:
|
||
|
dmc::MeshBuilder _mesh_builder;
|
||
|
dmc::DualGrid _dual_grid;
|
||
|
dmc::OctreeNodePool _octree_node_pool;
|
||
|
real_t _geometric_error = 0.1;
|
||
|
MeshMode _mesh_mode = MESH_NORMAL;
|
||
|
SimplifyMode _simplify_mode = SIMPLIFY_OCTREE_BOTTOM_UP;
|
||
|
SeamMode _seam_mode = SEAM_NONE;
|
||
|
|
||
|
struct Stats {
|
||
|
real_t octree_build_time = 0;
|
||
|
real_t dualgrid_derivation_time = 0;
|
||
|
real_t meshing_time = 0;
|
||
|
real_t commit_time = 0;
|
||
|
};
|
||
|
|
||
|
Stats _stats;
|
||
|
};
|
||
|
|
||
|
VARIANT_ENUM_CAST(VoxelMesherDMC::SimplifyMode)
|
||
|
VARIANT_ENUM_CAST(VoxelMesherDMC::MeshMode)
|
||
|
VARIANT_ENUM_CAST(VoxelMesherDMC::SeamMode)
|
||
|
|
||
|
#endif // VOXEL_MESHER_DMC_H
|