2019-09-03 22:54:40 +01:00
|
|
|
#include "voxel_tool.h"
|
2020-09-14 19:33:02 +01:00
|
|
|
#include "../storage/voxel_buffer.h"
|
2020-01-26 20:29:44 +00:00
|
|
|
#include "../terrain/voxel_lod_terrain.h"
|
2020-07-08 20:48:52 +01:00
|
|
|
#include "../util/macros.h"
|
2020-08-30 03:59:02 +01:00
|
|
|
#include "../util/profiling.h"
|
2019-09-03 22:54:40 +01:00
|
|
|
|
2019-09-08 19:42:25 +01:00
|
|
|
Vector3 VoxelRaycastResult::_b_get_position() const {
|
|
|
|
return position.to_vec3();
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector3 VoxelRaycastResult::_b_get_previous_position() const {
|
|
|
|
return previous_position.to_vec3();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelRaycastResult::_bind_methods() {
|
|
|
|
ClassDB::bind_method(D_METHOD("get_position"), &VoxelRaycastResult::_b_get_position);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_previous_position"), &VoxelRaycastResult::_b_get_previous_position);
|
|
|
|
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "position"), "", "get_position");
|
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::VECTOR3, "previous_position"), "", "get_previous_position");
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
void VoxelTool::set_value(uint64_t val) {
|
2019-09-03 22:54:40 +01:00
|
|
|
_value = val;
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
uint64_t VoxelTool::get_value() const {
|
2019-09-03 22:54:40 +01:00
|
|
|
return _value;
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
void VoxelTool::set_eraser_value(uint64_t value) {
|
2019-09-03 22:54:40 +01:00
|
|
|
_eraser_value = value;
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
uint64_t VoxelTool::get_eraser_value() const {
|
2019-09-03 22:54:40 +01:00
|
|
|
return _eraser_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelTool::set_channel(int channel) {
|
|
|
|
ERR_FAIL_INDEX(channel, VoxelBuffer::MAX_CHANNELS);
|
|
|
|
_channel = channel;
|
|
|
|
}
|
|
|
|
|
|
|
|
int VoxelTool::get_channel() const {
|
|
|
|
return _channel;
|
|
|
|
}
|
|
|
|
|
2019-09-05 20:47:08 +01:00
|
|
|
void VoxelTool::set_mode(Mode mode) {
|
|
|
|
_mode = mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
VoxelTool::Mode VoxelTool::get_mode() const {
|
|
|
|
return _mode;
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:58:36 +01:00
|
|
|
Ref<VoxelRaycastResult> VoxelTool::raycast(Vector3 pos, Vector3 dir, float max_distance, uint32_t collision_mask) {
|
2019-09-03 22:54:40 +01:00
|
|
|
ERR_PRINT("Not implemented");
|
2019-09-08 19:42:25 +01:00
|
|
|
return Ref<VoxelRaycastResult>();
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
uint64_t VoxelTool::get_voxel(Vector3i pos) {
|
2019-09-03 22:54:40 +01:00
|
|
|
return _get_voxel(pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
float VoxelTool::get_voxel_f(Vector3i pos) {
|
|
|
|
return _get_voxel_f(pos);
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
void VoxelTool::set_voxel(Vector3i pos, uint64_t v) {
|
2019-09-03 22:54:40 +01:00
|
|
|
Rect3i box(pos, Vector3i(1));
|
2019-09-06 23:24:56 +01:00
|
|
|
if (!is_area_editable(box)) {
|
2020-07-08 20:48:52 +01:00
|
|
|
PRINT_VERBOSE("Area not editable");
|
2019-09-06 23:24:56 +01:00
|
|
|
return;
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
2019-09-06 23:24:56 +01:00
|
|
|
_set_voxel(pos, v);
|
|
|
|
_post_edit(box);
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelTool::set_voxel_f(Vector3i pos, float v) {
|
|
|
|
Rect3i box(pos, Vector3i(1));
|
2019-09-06 23:24:56 +01:00
|
|
|
if (!is_area_editable(box)) {
|
2020-07-08 20:48:52 +01:00
|
|
|
PRINT_VERBOSE("Area not editable");
|
2019-09-06 23:24:56 +01:00
|
|
|
return;
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
2019-09-06 23:24:56 +01:00
|
|
|
_set_voxel_f(pos, v);
|
|
|
|
_post_edit(box);
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelTool::do_point(Vector3i pos) {
|
|
|
|
Rect3i box(pos, Vector3i(1));
|
|
|
|
if (!is_area_editable(box)) {
|
|
|
|
return;
|
|
|
|
}
|
2020-01-02 21:03:44 +00:00
|
|
|
if (_channel == VoxelBuffer::CHANNEL_SDF) {
|
2019-09-08 19:47:42 +01:00
|
|
|
_set_voxel_f(pos, _mode == MODE_REMOVE ? 1.0 : -1.0);
|
2019-09-03 22:54:40 +01:00
|
|
|
} else {
|
2019-09-08 19:47:42 +01:00
|
|
|
_set_voxel(pos, _mode == MODE_REMOVE ? _eraser_value : _value);
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
|
|
|
_post_edit(box);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelTool::do_line(Vector3i begin, Vector3i end) {
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelTool::do_circle(Vector3i pos, int radius, Vector3i direction) {
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
uint64_t VoxelTool::_get_voxel(Vector3i pos) {
|
2019-09-03 22:54:40 +01:00
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
float VoxelTool::_get_voxel_f(Vector3i pos) {
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
void VoxelTool::_set_voxel(Vector3i pos, uint64_t v) {
|
2019-09-03 22:54:40 +01:00
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelTool::_set_voxel_f(Vector3i pos, float v) {
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO May be worth using VoxelBuffer::read_write_action() in the future with a lambda,
|
|
|
|
// so we avoid the burden of going through get/set, validation and rehash access to blocks.
|
|
|
|
// Would work well by avoiding virtual as well using a specialized implementation.
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
inline float sdf_blend(float src_value, float dst_value, VoxelTool::Mode mode) {
|
|
|
|
float res;
|
|
|
|
switch (mode) {
|
|
|
|
case VoxelTool::MODE_ADD:
|
2020-01-21 20:00:29 +00:00
|
|
|
// Union
|
2019-09-03 22:54:40 +01:00
|
|
|
res = min(src_value, dst_value);
|
|
|
|
break;
|
|
|
|
|
2019-09-08 19:47:42 +01:00
|
|
|
case VoxelTool::MODE_REMOVE:
|
2020-01-21 20:00:29 +00:00
|
|
|
// Relative complement (or difference)
|
2019-12-31 16:37:40 +00:00
|
|
|
res = max(1.f - src_value, dst_value);
|
2019-09-03 22:54:40 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VoxelTool::MODE_SET:
|
|
|
|
res = src_value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
res = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
void VoxelTool::do_sphere(Vector3 center, float radius) {
|
2020-08-30 03:59:02 +01:00
|
|
|
VOXEL_PROFILE_SCOPE();
|
|
|
|
|
2019-09-11 21:25:15 +01:00
|
|
|
Rect3i box(Vector3i(center) - Vector3i(Math::floor(radius)), Vector3i(Math::ceil(radius) * 2));
|
2019-09-03 22:54:40 +01:00
|
|
|
|
|
|
|
if (!is_area_editable(box)) {
|
2020-07-08 20:48:52 +01:00
|
|
|
PRINT_VERBOSE("Area not editable");
|
2019-09-03 22:54:40 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-02 21:03:44 +00:00
|
|
|
if (_channel == VoxelBuffer::CHANNEL_SDF) {
|
2019-09-03 22:54:40 +01:00
|
|
|
box.for_each_cell([this, center, radius](Vector3i pos) {
|
|
|
|
float d = pos.to_vec3().distance_to(center) - radius;
|
|
|
|
_set_voxel_f(pos, sdf_blend(d, get_voxel_f(pos), _mode));
|
|
|
|
});
|
|
|
|
|
|
|
|
} else {
|
2019-09-08 19:47:42 +01:00
|
|
|
int value = _mode == MODE_REMOVE ? _eraser_value : _value;
|
2019-09-03 22:54:40 +01:00
|
|
|
|
|
|
|
box.for_each_cell([this, center, radius, value](Vector3i pos) {
|
|
|
|
float d = pos.to_vec3().distance_to(center);
|
|
|
|
if (d <= radius) {
|
2019-09-11 21:30:26 +01:00
|
|
|
_set_voxel(pos, value);
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
_post_edit(box);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelTool::do_box(Vector3i begin, Vector3i end) {
|
2020-08-30 03:59:02 +01:00
|
|
|
VOXEL_PROFILE_SCOPE();
|
2020-08-05 17:21:13 -07:00
|
|
|
Vector3i::sort_min_max(begin, end);
|
2020-08-29 22:57:27 +01:00
|
|
|
Rect3i box = Rect3i::from_min_max(begin, end + Vector3i(1, 1, 1));
|
2020-08-05 17:21:13 -07:00
|
|
|
|
|
|
|
if (!is_area_editable(box)) {
|
|
|
|
PRINT_VERBOSE("Area not editable");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_channel == VoxelBuffer::CHANNEL_SDF) {
|
|
|
|
|
|
|
|
box.for_each_cell([this](Vector3i pos) {
|
|
|
|
_set_voxel_f(pos, sdf_blend(-1.0, get_voxel_f(pos), _mode));
|
|
|
|
});
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
int value = _mode == MODE_REMOVE ? _eraser_value : _value;
|
|
|
|
|
|
|
|
box.for_each_cell([this, value](Vector3i pos) {
|
|
|
|
_set_voxel(pos, value);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
_post_edit(box);
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
|
|
|
|
2020-08-30 18:51:20 +01:00
|
|
|
void VoxelTool::copy(Vector3i pos, Ref<VoxelBuffer> dst, uint64_t mask_value) {
|
|
|
|
ERR_FAIL_COND(dst.is_null());
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:54:47 +01:00
|
|
|
void VoxelTool::paste(Vector3i p_pos, Ref<VoxelBuffer> p_voxels, uint64_t mask_value) {
|
2019-09-03 22:54:40 +01:00
|
|
|
ERR_FAIL_COND(p_voxels.is_null());
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VoxelTool::is_area_editable(const Rect3i &box) const {
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelTool::_post_edit(const Rect3i &box) {
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
}
|
|
|
|
|
2020-08-10 19:03:01 +01:00
|
|
|
void VoxelTool::set_voxel_metadata(Vector3i pos, Variant meta) {
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
}
|
|
|
|
|
|
|
|
Variant VoxelTool::get_voxel_metadata(Vector3i pos) {
|
|
|
|
ERR_PRINT("Not implemented");
|
|
|
|
return Variant();
|
|
|
|
}
|
|
|
|
|
2019-09-03 22:54:40 +01:00
|
|
|
void VoxelTool::_bind_methods() {
|
|
|
|
ClassDB::bind_method(D_METHOD("set_value", "v"), &VoxelTool::set_value);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_value"), &VoxelTool::get_value);
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("set_channel", "v"), &VoxelTool::set_channel);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_channel"), &VoxelTool::get_channel);
|
|
|
|
|
2019-09-05 20:47:08 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("set_mode", "m"), &VoxelTool::set_mode);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_mode"), &VoxelTool::get_mode);
|
|
|
|
|
2019-09-03 22:54:40 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("set_eraser_value", "v"), &VoxelTool::set_eraser_value);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_eraser_value"), &VoxelTool::get_eraser_value);
|
|
|
|
|
|
|
|
ClassDB::bind_method(D_METHOD("get_voxel", "pos"), &VoxelTool::_b_get_voxel);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_voxel_f", "pos"), &VoxelTool::_b_get_voxel_f);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_voxel", "pos", "v"), &VoxelTool::_b_set_voxel);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_voxel_f", "pos", "v"), &VoxelTool::_b_set_voxel_f);
|
|
|
|
ClassDB::bind_method(D_METHOD("do_point", "pos"), &VoxelTool::_b_do_point);
|
2020-08-29 22:57:27 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("do_sphere", "center", "radius"), &VoxelTool::_b_do_sphere);
|
2020-08-05 17:21:13 -07:00
|
|
|
ClassDB::bind_method(D_METHOD("do_box", "begin", "end"), &VoxelTool::_b_do_box);
|
2019-09-08 19:42:25 +01:00
|
|
|
|
2020-08-10 19:03:01 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("set_voxel_metadata", "pos", "meta"), &VoxelTool::_b_set_voxel_metadata);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_voxel_metadata", "pos"), &VoxelTool::_b_get_voxel_metadata);
|
|
|
|
|
2020-08-04 23:54:40 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("paste", "dst_pos", "src_buffer", "src_mask_value"), &VoxelTool::_b_paste);
|
|
|
|
|
2020-08-06 19:58:36 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("raycast", "origin", "direction", "max_distance", "collision_mask"),
|
|
|
|
&VoxelTool::_b_raycast, DEFVAL(10.0), DEFVAL(0xffffffff));
|
2019-09-03 22:54:40 +01:00
|
|
|
|
2020-08-24 13:51:08 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("is_area_editable", "box"), &VoxelTool::_b_is_area_editable);
|
|
|
|
|
2019-09-03 22:54:40 +01:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "value"), "set_value", "get_value");
|
2019-09-05 20:47:08 +01:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "channel", PROPERTY_HINT_ENUM, VoxelBuffer::CHANNEL_ID_HINT_STRING), "set_channel", "get_channel");
|
2019-09-03 22:54:40 +01:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "eraser_value"), "set_eraser_value", "get_eraser_value");
|
2019-09-05 20:47:08 +01:00
|
|
|
ADD_PROPERTY(PropertyInfo(Variant::INT, "mode", PROPERTY_HINT_ENUM, "Add,Remove,Set"), "set_mode", "get_mode");
|
2019-09-03 22:54:40 +01:00
|
|
|
|
|
|
|
BIND_ENUM_CONSTANT(MODE_ADD);
|
2019-09-08 19:47:42 +01:00
|
|
|
BIND_ENUM_CONSTANT(MODE_REMOVE);
|
2019-09-03 22:54:40 +01:00
|
|
|
BIND_ENUM_CONSTANT(MODE_SET);
|
|
|
|
}
|