#include "voxel_isosurface_tool.h" #include "util/utility.h" VoxelIsoSurfaceTool::VoxelIsoSurfaceTool() { _iso_scale = 1.0; } void VoxelIsoSurfaceTool::set_buffer(Ref buffer) { _buffer = buffer; } Ref VoxelIsoSurfaceTool::get_buffer() const { return _buffer; } void VoxelIsoSurfaceTool::set_iso_scale(float iso_scale) { _iso_scale = iso_scale; } float VoxelIsoSurfaceTool::get_iso_scale() const { return _iso_scale; } void VoxelIsoSurfaceTool::set_offset(Vector3 offset) { _offset = offset; } Vector3 VoxelIsoSurfaceTool::get_offset() const { return _offset; } namespace { inline void do_op(VoxelBuffer &buffer, int x, int y, int z, float d1, VoxelIsoSurfaceTool::Operation op) { float res; switch (op) { case VoxelIsoSurfaceTool::OP_ADD: res = MIN(d1, buffer.get_voxel_f(x, y, z, VoxelBuffer::CHANNEL_ISOLEVEL)); break; case VoxelIsoSurfaceTool::OP_SUBTRACT: res = MAX(1.0 - d1, buffer.get_voxel_f(x, y, z, VoxelBuffer::CHANNEL_ISOLEVEL)); break; case VoxelIsoSurfaceTool::OP_SET: res = d1; break; default: res = 0; break; } buffer.set_voxel_f(res, x, y, z, VoxelBuffer::CHANNEL_ISOLEVEL); } } // namespace void VoxelIsoSurfaceTool::do_sphere(Vector3 center, real_t radius, Operation op) { ERR_FAIL_COND(_buffer.is_null()); VoxelBuffer &buffer = **_buffer; center += _offset; for (int z = 0; z < buffer.get_size().z; ++z) { for (int x = 0; x < buffer.get_size().x; ++x) { for (int y = 0; y < buffer.get_size().y; ++y) { float d1 = (center.distance_to(Vector3(x, y, z)) - radius) * _iso_scale; do_op(buffer, x, y, z, d1, op); } } } } void VoxelIsoSurfaceTool::do_plane(Plane plane, Operation op) { ERR_FAIL_COND(_buffer.is_null()); VoxelBuffer &buffer = **_buffer; plane = Plane(plane.center() + _offset, plane.normal); for (int z = 0; z < buffer.get_size().z; ++z) { for (int x = 0; x < buffer.get_size().x; ++x) { for (int y = 0; y < buffer.get_size().y; ++y) { float d1 = plane.distance_to(Vector3(x, y, z)) * _iso_scale; do_op(buffer, x, y, z, d1, op); } } } } namespace { inline float sdf_cube(Vector3 pos, Vector3 extents) { Vector3 d = pos.abs() - extents; return min(max(d.x, max(d.y, d.z)), 0.f) + Vector3(max(d.x, 0.f), max(d.y, 0.f), max(d.z, 0.f)).length(); } } // namespace void VoxelIsoSurfaceTool::do_cube(Transform transform, Vector3 extents, Operation op) { ERR_FAIL_COND(_buffer.is_null()); VoxelBuffer &buffer = **_buffer; transform.origin += _offset; Transform inv_transform = transform.affine_inverse(); for (int z = 0; z < buffer.get_size().z; ++z) { for (int x = 0; x < buffer.get_size().x; ++x) { for (int y = 0; y < buffer.get_size().y; ++y) { Vector3 pos = inv_transform.xform(Vector3(x, y, z)); do_op(buffer, x, y, z, sdf_cube(pos, extents) * _iso_scale, op); } } } } void VoxelIsoSurfaceTool::do_heightmap(Ref heightmap, Vector3 offset, real_t vertical_scale, Operation op) { ERR_FAIL_COND(_buffer.is_null()); VoxelBuffer &buffer = **_buffer; ERR_FAIL_COND(heightmap.is_null()); Image &im = **heightmap; offset += _offset; im.lock(); for (int z = 0; z < buffer.get_size().z; ++z) { for (int x = 0; x < buffer.get_size().x; ++x) { int px = int(Math::fmod(x + offset.x, im.get_width())); int py = int(Math::fmod(z + offset.z, im.get_height())); float h = im.get_pixel(px, py).r * vertical_scale + offset.y; for (int y = 0; y < buffer.get_size().y; ++y) { // Not a true distance to heightmap, but might be enough float d = (y - h) * _iso_scale; do_op(buffer, x, y, z, d, op); } } } im.unlock(); } void VoxelIsoSurfaceTool::_bind_methods() { ClassDB::bind_method(D_METHOD("set_buffer", "voxel_buffer"), &VoxelIsoSurfaceTool::set_buffer); ClassDB::bind_method(D_METHOD("get_buffer"), &VoxelIsoSurfaceTool::get_buffer); ClassDB::bind_method(D_METHOD("set_iso_scale", "iso_scale"), &VoxelIsoSurfaceTool::set_iso_scale); ClassDB::bind_method(D_METHOD("get_iso_scale"), &VoxelIsoSurfaceTool::get_iso_scale); ClassDB::bind_method(D_METHOD("set_offset", "offset"), &VoxelIsoSurfaceTool::set_offset); ClassDB::bind_method(D_METHOD("get_offset"), &VoxelIsoSurfaceTool::get_offset); ClassDB::bind_method(D_METHOD("do_sphere", "center", "radius", "op"), &VoxelIsoSurfaceTool::do_sphere, DEFVAL(OP_ADD)); ClassDB::bind_method(D_METHOD("do_plane", "plane", "op"), &VoxelIsoSurfaceTool::do_plane, DEFVAL(OP_ADD)); ClassDB::bind_method(D_METHOD("do_cube", "transform", "extents", "op"), &VoxelIsoSurfaceTool::do_cube, DEFVAL(OP_ADD)); ClassDB::bind_method(D_METHOD("do_heightmap", "heightmap", "offset", "vertical_scale", "op"), &VoxelIsoSurfaceTool::do_heightmap, DEFVAL(OP_ADD)); BIND_ENUM_CONSTANT(OP_ADD); BIND_ENUM_CONSTANT(OP_SUBTRACT); BIND_ENUM_CONSTANT(OP_SET); }