2020-01-02 20:31:05 +00:00
|
|
|
#define VOXEL_BUFFER_USE_MEMORY_POOL
|
|
|
|
|
|
|
|
#ifdef VOXEL_BUFFER_USE_MEMORY_POOL
|
|
|
|
#include "voxel_memory_pool.h"
|
|
|
|
#endif
|
|
|
|
|
2020-01-26 20:29:44 +00:00
|
|
|
#include "edition/voxel_tool_buffer.h"
|
2016-05-01 15:00:02 +02:00
|
|
|
#include "voxel_buffer.h"
|
2018-09-30 01:55:09 +01:00
|
|
|
|
2020-08-02 18:57:08 +01:00
|
|
|
#include <core/image.h>
|
2020-01-04 19:20:49 +00:00
|
|
|
#include <core/io/marshalls.h>
|
2018-09-19 20:25:04 +01:00
|
|
|
#include <core/math/math_funcs.h>
|
2017-08-13 01:19:39 +02:00
|
|
|
#include <string.h>
|
2016-05-01 15:00:02 +02:00
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
inline uint8_t *allocate_channel_data(uint32_t size) {
|
|
|
|
#ifdef VOXEL_BUFFER_USE_MEMORY_POOL
|
|
|
|
return VoxelMemoryPool::get_singleton()->allocate(size);
|
|
|
|
#else
|
|
|
|
return (uint8_t *)memalloc(size * sizeof(uint8_t));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void free_channel_data(uint8_t *data, uint32_t size) {
|
|
|
|
#ifdef VOXEL_BUFFER_USE_MEMORY_POOL
|
|
|
|
VoxelMemoryPool::get_singleton()->recycle(data, size);
|
|
|
|
#else
|
|
|
|
memfree(data);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t g_depth_bit_counts[] = {
|
2020-01-25 20:45:47 +00:00
|
|
|
8, 16, 32, 64
|
2020-01-04 19:20:49 +00:00
|
|
|
};
|
|
|
|
uint64_t g_depth_max_values[] = {
|
|
|
|
0xff, // 8
|
|
|
|
0xffff, // 16
|
|
|
|
0xffffffff, // 32
|
|
|
|
0xffffffffffffffff // 64
|
|
|
|
};
|
|
|
|
|
|
|
|
inline uint32_t get_depth_bit_count(VoxelBuffer::Depth d) {
|
2020-02-07 04:52:06 +01:00
|
|
|
CRASH_COND(d < 0 || d >= VoxelBuffer::DEPTH_COUNT);
|
2020-01-04 19:20:49 +00:00
|
|
|
return g_depth_bit_counts[d];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline uint64_t get_max_value_for_depth(VoxelBuffer::Depth d) {
|
2020-02-07 04:52:06 +01:00
|
|
|
CRASH_COND(d < 0 || d >= VoxelBuffer::DEPTH_COUNT);
|
2020-01-04 19:20:49 +00:00
|
|
|
return g_depth_max_values[d];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline uint64_t clamp_value_for_depth(uint64_t value, VoxelBuffer::Depth d) {
|
|
|
|
uint64_t max_val = get_max_value_for_depth(d);
|
|
|
|
if (value >= max_val) {
|
|
|
|
return max_val;
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static_assert(sizeof(uint32_t) == sizeof(float), "uint32_t and float cannot be marshalled back and forth");
|
|
|
|
static_assert(sizeof(uint64_t) == sizeof(double), "uint64_t and double cannot be marshalled back and forth");
|
|
|
|
|
|
|
|
inline uint64_t real_to_raw_voxel(real_t value, VoxelBuffer::Depth depth) {
|
|
|
|
switch (depth) {
|
|
|
|
case VoxelBuffer::DEPTH_8_BIT:
|
|
|
|
return clamp(static_cast<int>(128.f * value + 128.f), 0, 0xff);
|
|
|
|
case VoxelBuffer::DEPTH_16_BIT:
|
|
|
|
return clamp(static_cast<int>(0x7fff * value + 0x7fff), 0, 0xffff);
|
|
|
|
case VoxelBuffer::DEPTH_32_BIT: {
|
|
|
|
MarshallFloat m;
|
|
|
|
m.f = value;
|
|
|
|
return m.i;
|
|
|
|
}
|
|
|
|
case VoxelBuffer::DEPTH_64_BIT: {
|
|
|
|
MarshallDouble m;
|
|
|
|
m.d = value;
|
|
|
|
return m.l;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
CRASH_NOW();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-19 01:22:48 +00:00
|
|
|
inline real_t raw_voxel_to_real(uint64_t value, VoxelBuffer::Depth depth) {
|
2020-01-04 19:20:49 +00:00
|
|
|
// Depths below 32 are normalized between -1 and 1
|
|
|
|
switch (depth) {
|
|
|
|
case VoxelBuffer::DEPTH_8_BIT:
|
|
|
|
return (static_cast<real_t>(value) - 0x7f) / 0x7f;
|
|
|
|
case VoxelBuffer::DEPTH_16_BIT:
|
|
|
|
return (static_cast<real_t>(value) - 0x7fff) / 0x7fff;
|
|
|
|
case VoxelBuffer::DEPTH_32_BIT: {
|
|
|
|
MarshallFloat m;
|
|
|
|
m.i = value;
|
|
|
|
return m.f;
|
|
|
|
}
|
|
|
|
case VoxelBuffer::DEPTH_64_BIT: {
|
|
|
|
MarshallDouble m;
|
2020-02-05 16:37:39 +01:00
|
|
|
m.l = value;
|
|
|
|
return m.d;
|
2020-01-04 19:20:49 +00:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
CRASH_NOW();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2019-05-30 00:46:08 +01:00
|
|
|
const char *VoxelBuffer::CHANNEL_ID_HINT_STRING = "Type,Sdf,Data2,Data3,Data4,Data5,Data6,Data7";
|
|
|
|
|
2016-05-01 15:00:02 +02:00
|
|
|
VoxelBuffer::VoxelBuffer() {
|
2020-01-02 21:03:44 +00:00
|
|
|
_channels[CHANNEL_SDF].defval = 255;
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
VoxelBuffer::~VoxelBuffer() {
|
2017-01-01 04:40:16 +01:00
|
|
|
clear();
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelBuffer::create(int sx, int sy, int sz) {
|
2017-01-01 04:40:16 +01:00
|
|
|
if (sx <= 0 || sy <= 0 || sz <= 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Vector3i new_size(sx, sy, sz);
|
|
|
|
if (new_size != _size) {
|
|
|
|
for (unsigned int i = 0; i < MAX_CHANNELS; ++i) {
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[i];
|
2017-01-01 04:40:16 +01:00
|
|
|
if (channel.data) {
|
2019-04-22 20:15:19 +01:00
|
|
|
// Channel already contained data
|
2017-01-05 02:39:40 +01:00
|
|
|
delete_channel(i);
|
2019-04-22 20:15:19 +01:00
|
|
|
create_channel(i, new_size, channel.defval);
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_size = new_size;
|
|
|
|
}
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2019-09-06 23:24:56 +01:00
|
|
|
void VoxelBuffer::create(Vector3i size) {
|
|
|
|
create(size.x, size.y, size.z);
|
|
|
|
}
|
|
|
|
|
2016-05-01 15:00:02 +02:00
|
|
|
void VoxelBuffer::clear() {
|
2017-01-01 04:40:16 +01:00
|
|
|
for (unsigned int i = 0; i < MAX_CHANNELS; ++i) {
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[i];
|
2017-01-01 04:40:16 +01:00
|
|
|
if (channel.data) {
|
2017-01-05 02:39:40 +01:00
|
|
|
delete_channel(i);
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
}
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::clear_channel(unsigned int channel_index, uint64_t clear_value) {
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
2020-01-04 19:20:49 +00:00
|
|
|
Channel &channel = _channels[channel_index];
|
|
|
|
if (channel.data) {
|
2017-01-05 02:39:40 +01:00
|
|
|
delete_channel(channel_index);
|
2019-04-21 19:31:35 +01:00
|
|
|
}
|
2020-01-04 19:20:49 +00:00
|
|
|
channel.defval = clamp_value_for_depth(clear_value, channel.depth);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelBuffer::clear_channel_f(unsigned int channel_index, real_t clear_value) {
|
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
|
|
|
const Channel &channel = _channels[channel_index];
|
|
|
|
clear_channel(channel_index, real_to_raw_voxel(clear_value, channel.depth));
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::set_default_values(FixedArray<uint64_t, VoxelBuffer::MAX_CHANNELS> values) {
|
2017-08-13 01:19:39 +02:00
|
|
|
for (unsigned int i = 0; i < MAX_CHANNELS; ++i) {
|
2020-01-04 19:20:49 +00:00
|
|
|
_channels[i].defval = clamp_value_for_depth(values[i], _channels[i].depth);
|
2017-01-05 02:39:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
uint64_t VoxelBuffer::get_voxel(int x, int y, int z, unsigned int channel_index) const {
|
2017-01-05 02:39:40 +01:00
|
|
|
ERR_FAIL_INDEX_V(channel_index, MAX_CHANNELS, 0);
|
2017-01-01 04:40:16 +01:00
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
const Channel &channel = _channels[channel_index];
|
2017-01-01 04:40:16 +01:00
|
|
|
|
|
|
|
if (validate_pos(x, y, z) && channel.data) {
|
2020-01-04 19:20:49 +00:00
|
|
|
uint32_t i = index(x, y, z);
|
|
|
|
|
|
|
|
switch (channel.depth) {
|
|
|
|
case DEPTH_8_BIT:
|
|
|
|
return channel.data[i];
|
|
|
|
|
|
|
|
case DEPTH_16_BIT:
|
|
|
|
return ((uint16_t *)channel.data)[i];
|
|
|
|
|
|
|
|
case DEPTH_32_BIT:
|
|
|
|
return ((uint32_t *)channel.data)[i];
|
|
|
|
|
|
|
|
case DEPTH_64_BIT:
|
|
|
|
return ((uint64_t *)channel.data)[i];
|
|
|
|
|
|
|
|
default:
|
|
|
|
CRASH_NOW();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
return channel.data[index(x, y, z)];
|
2020-01-04 19:20:49 +00:00
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
} else {
|
2017-01-01 04:40:16 +01:00
|
|
|
return channel.defval;
|
|
|
|
}
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::set_voxel(uint64_t value, int x, int y, int z, unsigned int channel_index) {
|
2017-08-13 01:19:39 +02:00
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_COND(!validate_pos(x, y, z));
|
2016-05-01 15:00:02 +02:00
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[channel_index];
|
2016-05-01 15:00:02 +02:00
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
value = clamp_value_for_depth(value, channel.depth);
|
|
|
|
bool do_set = true;
|
|
|
|
|
2020-07-25 23:19:08 +01:00
|
|
|
if (channel.data == nullptr) {
|
2017-03-29 22:36:42 +02:00
|
|
|
if (channel.defval != value) {
|
2019-04-22 20:15:19 +01:00
|
|
|
// Allocate channel with same initial values as defval
|
|
|
|
create_channel(channel_index, _size, channel.defval);
|
2020-01-04 19:20:49 +00:00
|
|
|
} else {
|
|
|
|
do_set = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (do_set) {
|
|
|
|
uint32_t i = index(x, y, z);
|
|
|
|
|
|
|
|
switch (channel.depth) {
|
|
|
|
case DEPTH_8_BIT:
|
|
|
|
channel.data[i] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_16_BIT:
|
|
|
|
((uint16_t *)channel.data)[i] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_32_BIT:
|
|
|
|
((uint32_t *)channel.data)[i] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_64_BIT:
|
|
|
|
((uint64_t *)channel.data)[i] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
CRASH_NOW();
|
|
|
|
break;
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
}
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
real_t VoxelBuffer::get_voxel_f(int x, int y, int z, unsigned int channel_index) const {
|
|
|
|
ERR_FAIL_INDEX_V(channel_index, MAX_CHANNELS, 0);
|
|
|
|
return raw_voxel_to_real(get_voxel(x, y, z, channel_index), _channels[channel_index].depth);
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoxelBuffer::set_voxel_f(real_t value, int x, int y, int z, unsigned int channel_index) {
|
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
|
|
|
set_voxel(real_to_raw_voxel(value, _channels[channel_index].depth), x, y, z, channel_index);
|
|
|
|
}
|
|
|
|
|
2018-10-04 23:13:03 +01:00
|
|
|
// This version does not cause errors if out of bounds. Use only if it's okay to be outside.
|
|
|
|
void VoxelBuffer::try_set_voxel(int x, int y, int z, int value, unsigned int channel_index) {
|
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
2019-04-21 19:31:35 +01:00
|
|
|
if (!validate_pos(x, y, z)) {
|
2018-10-04 23:13:03 +01:00
|
|
|
return;
|
2019-04-21 19:31:35 +01:00
|
|
|
}
|
2020-01-04 19:20:49 +00:00
|
|
|
set_voxel(x, y, z, value, channel_index);
|
2018-10-04 23:13:03 +01:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::fill(uint64_t defval, unsigned int channel_index) {
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
2016-05-05 21:08:52 +02:00
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[channel_index];
|
2020-01-04 19:20:49 +00:00
|
|
|
|
|
|
|
defval = clamp_value_for_depth(defval, channel.depth);
|
|
|
|
|
2020-07-25 23:19:08 +01:00
|
|
|
if (channel.data == nullptr) {
|
2018-09-02 18:48:08 +01:00
|
|
|
// Channel is already optimized and uniform
|
2019-04-20 17:32:10 +01:00
|
|
|
if (channel.defval == defval) {
|
2018-09-02 18:48:08 +01:00
|
|
|
// No change
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
// Just change default value
|
|
|
|
channel.defval = defval;
|
|
|
|
return;
|
|
|
|
}
|
2019-04-21 19:31:35 +01:00
|
|
|
}
|
2016-05-05 21:08:52 +02:00
|
|
|
|
2017-01-01 04:40:16 +01:00
|
|
|
unsigned int volume = get_volume();
|
2020-01-04 19:20:49 +00:00
|
|
|
|
|
|
|
switch (channel.depth) {
|
|
|
|
|
|
|
|
case DEPTH_8_BIT:
|
|
|
|
memset(channel.data, defval, channel.size_in_bytes);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_16_BIT:
|
|
|
|
for (uint32_t i = 0; i < volume; ++i) {
|
|
|
|
((uint16_t *)channel.data)[i] = defval;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_32_BIT:
|
|
|
|
for (uint32_t i = 0; i < volume; ++i) {
|
|
|
|
((uint32_t *)channel.data)[i] = defval;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_64_BIT:
|
|
|
|
for (uint32_t i = 0; i < volume; ++i) {
|
|
|
|
((uint64_t *)channel.data)[i] = defval;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
CRASH_NOW();
|
|
|
|
break;
|
|
|
|
}
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::fill_area(uint64_t defval, Vector3i min, Vector3i max, unsigned int channel_index) {
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
|
|
|
|
|
|
|
Vector3i::sort_min_max(min, max);
|
|
|
|
|
2018-10-04 23:13:03 +01:00
|
|
|
min.clamp_to(Vector3i(0, 0, 0), _size + Vector3i(1, 1, 1));
|
2017-08-13 01:19:39 +02:00
|
|
|
max.clamp_to(Vector3i(0, 0, 0), _size + Vector3i(1, 1, 1));
|
2017-01-01 04:40:16 +01:00
|
|
|
Vector3i area_size = max - min;
|
|
|
|
|
2019-04-21 19:31:35 +01:00
|
|
|
if (area_size.x == 0 || area_size.y == 0 || area_size.z == 0) {
|
2018-10-04 23:13:03 +01:00
|
|
|
return;
|
2019-04-21 19:31:35 +01:00
|
|
|
}
|
2018-10-04 23:13:03 +01:00
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[channel_index];
|
2020-01-04 19:20:49 +00:00
|
|
|
defval = clamp_value_for_depth(defval, channel.depth);
|
|
|
|
|
2020-07-25 23:19:08 +01:00
|
|
|
if (channel.data == nullptr) {
|
2019-04-21 19:31:35 +01:00
|
|
|
if (channel.defval == defval) {
|
2017-01-01 04:40:16 +01:00
|
|
|
return;
|
2019-04-21 19:31:35 +01:00
|
|
|
} else {
|
2019-04-22 20:15:19 +01:00
|
|
|
create_channel(channel_index, _size, channel.defval);
|
2019-04-21 19:31:35 +01:00
|
|
|
}
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Vector3i pos;
|
2019-06-18 14:24:56 +09:00
|
|
|
unsigned int volume = get_volume();
|
2017-01-01 04:40:16 +01:00
|
|
|
for (pos.z = min.z; pos.z < max.z; ++pos.z) {
|
|
|
|
for (pos.x = min.x; pos.x < max.x; ++pos.x) {
|
2020-01-04 19:20:49 +00:00
|
|
|
|
2017-01-01 04:40:16 +01:00
|
|
|
unsigned int dst_ri = index(pos.x, pos.y + min.y, pos.z);
|
2018-10-04 23:13:03 +01:00
|
|
|
CRASH_COND(dst_ri >= volume);
|
2020-01-04 19:20:49 +00:00
|
|
|
|
|
|
|
switch (channel.depth) {
|
|
|
|
|
|
|
|
case DEPTH_8_BIT:
|
|
|
|
// Fill row by row
|
|
|
|
memset(&channel.data[dst_ri], defval, area_size.y * sizeof(uint8_t));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_16_BIT:
|
2020-02-13 21:49:11 +08:00
|
|
|
for (int i = 0; i < area_size.y; ++i) {
|
2020-01-04 19:20:49 +00:00
|
|
|
((uint16_t *)channel.data)[dst_ri + i] = defval;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_32_BIT:
|
2020-02-13 21:49:11 +08:00
|
|
|
for (int i = 0; i < area_size.y; ++i) {
|
2020-01-04 19:20:49 +00:00
|
|
|
((uint32_t *)channel.data)[dst_ri + i] = defval;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DEPTH_64_BIT:
|
2020-02-13 21:49:11 +08:00
|
|
|
for (int i = 0; i < area_size.y; ++i) {
|
2020-01-04 19:20:49 +00:00
|
|
|
((uint64_t *)channel.data)[dst_ri + i] = defval;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
CRASH_NOW();
|
|
|
|
break;
|
|
|
|
}
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
}
|
2016-05-05 21:08:52 +02:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::fill_f(real_t value, unsigned int channel) {
|
|
|
|
ERR_FAIL_INDEX(channel, MAX_CHANNELS);
|
|
|
|
fill(real_to_raw_voxel(value, _channels[channel].depth), channel);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
inline bool is_uniform(const uint8_t *p_data, uint32_t size) {
|
|
|
|
const T *data = (const T *)p_data;
|
|
|
|
T v0 = data[0];
|
|
|
|
for (unsigned int i = 1; i < size; ++i) {
|
|
|
|
if (data[i] != v0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-04-06 23:44:11 +02:00
|
|
|
bool VoxelBuffer::is_uniform(unsigned int channel_index) const {
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_INDEX_V(channel_index, MAX_CHANNELS, true);
|
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
const Channel &channel = _channels[channel_index];
|
2020-01-04 19:20:49 +00:00
|
|
|
if (channel.data == nullptr) {
|
2017-08-28 01:47:38 +02:00
|
|
|
// Channel has been optimized
|
2017-01-01 04:40:16 +01:00
|
|
|
return true;
|
2019-04-21 19:31:35 +01:00
|
|
|
}
|
2017-01-01 04:40:16 +01:00
|
|
|
|
|
|
|
unsigned int volume = get_volume();
|
2020-01-04 19:20:49 +00:00
|
|
|
|
|
|
|
// Channel isn't optimized, so must look at each voxel
|
|
|
|
switch (channel.depth) {
|
|
|
|
case DEPTH_8_BIT:
|
|
|
|
return ::is_uniform<uint8_t>(channel.data, volume);
|
|
|
|
case DEPTH_16_BIT:
|
|
|
|
return ::is_uniform<uint16_t>(channel.data, volume);
|
|
|
|
case DEPTH_32_BIT:
|
|
|
|
return ::is_uniform<uint32_t>(channel.data, volume);
|
|
|
|
case DEPTH_64_BIT:
|
|
|
|
return ::is_uniform<uint64_t>(channel.data, volume);
|
|
|
|
default:
|
|
|
|
CRASH_NOW();
|
|
|
|
break;
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2019-05-25 15:51:59 +01:00
|
|
|
void VoxelBuffer::compress_uniform_channels() {
|
2017-01-01 04:40:16 +01:00
|
|
|
for (unsigned int i = 0; i < MAX_CHANNELS; ++i) {
|
|
|
|
if (_channels[i].data && is_uniform(i)) {
|
|
|
|
clear_channel(i, _channels[i].data[0]);
|
|
|
|
}
|
|
|
|
}
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2019-05-28 00:40:09 +01:00
|
|
|
void VoxelBuffer::decompress_channel(unsigned int channel_index) {
|
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
2019-09-03 22:54:40 +01:00
|
|
|
Channel &channel = _channels[channel_index];
|
2019-05-28 00:40:09 +01:00
|
|
|
if (channel.data == nullptr) {
|
|
|
|
create_channel(channel_index, _size, channel.defval);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
VoxelBuffer::Compression VoxelBuffer::get_channel_compression(unsigned int channel_index) const {
|
|
|
|
ERR_FAIL_INDEX_V(channel_index, MAX_CHANNELS, VoxelBuffer::COMPRESSION_NONE);
|
2019-09-03 22:54:40 +01:00
|
|
|
const Channel &channel = _channels[channel_index];
|
2019-05-28 00:40:09 +01:00
|
|
|
if (channel.data == nullptr) {
|
|
|
|
return COMPRESSION_UNIFORM;
|
|
|
|
}
|
|
|
|
return COMPRESSION_NONE;
|
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::copy_from(const VoxelBuffer &other) {
|
|
|
|
// Copy all channels, assuming sizes and formats match
|
|
|
|
for (unsigned int i = 0; i < MAX_CHANNELS; ++i) {
|
|
|
|
copy_from(other, i);
|
2019-05-28 00:40:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
void VoxelBuffer::copy_from(const VoxelBuffer &other, unsigned int channel_index) {
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
2019-08-24 23:08:04 +01:00
|
|
|
ERR_FAIL_COND(other._size != _size);
|
2017-01-01 04:40:16 +01:00
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[channel_index];
|
|
|
|
const Channel &other_channel = other._channels[channel_index];
|
2017-01-01 04:40:16 +01:00
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
ERR_FAIL_COND(other_channel.depth != channel.depth);
|
|
|
|
|
2017-01-01 04:40:16 +01:00
|
|
|
if (other_channel.data) {
|
2020-07-25 23:19:08 +01:00
|
|
|
if (channel.data == nullptr) {
|
2017-01-01 04:40:16 +01:00
|
|
|
create_channel_noinit(channel_index, _size);
|
|
|
|
}
|
2020-01-26 01:33:45 +00:00
|
|
|
CRASH_COND(channel.size_in_bytes != other_channel.size_in_bytes);
|
|
|
|
memcpy(channel.data, other_channel.data, channel.size_in_bytes);
|
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
} else if (channel.data) {
|
2017-01-05 02:39:40 +01:00
|
|
|
delete_channel(channel_index);
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
channel.defval = other_channel.defval;
|
2020-01-04 19:20:49 +00:00
|
|
|
channel.depth = other_channel.depth;
|
2016-05-05 21:08:52 +02:00
|
|
|
}
|
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
void VoxelBuffer::copy_from(const VoxelBuffer &other, Vector3i src_min, Vector3i src_max, Vector3i dst_min, unsigned int channel_index) {
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[channel_index];
|
|
|
|
const Channel &other_channel = other._channels[channel_index];
|
2017-01-01 04:40:16 +01:00
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
ERR_FAIL_COND(other_channel.depth != channel.depth);
|
|
|
|
|
2019-08-20 20:50:09 +01:00
|
|
|
if (channel.data == nullptr && other_channel.data == nullptr && channel.defval == other_channel.defval) {
|
|
|
|
// No action needed
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-01-01 04:40:16 +01:00
|
|
|
Vector3i::sort_min_max(src_min, src_max);
|
|
|
|
|
|
|
|
src_min.clamp_to(Vector3i(0, 0, 0), other._size);
|
2017-08-13 01:19:39 +02:00
|
|
|
src_max.clamp_to(Vector3i(0, 0, 0), other._size + Vector3i(1, 1, 1));
|
2017-01-01 04:40:16 +01:00
|
|
|
|
|
|
|
dst_min.clamp_to(Vector3i(0, 0, 0), _size);
|
|
|
|
Vector3i area_size = src_max - src_min;
|
|
|
|
//Vector3i dst_max = dst_min + area_size;
|
|
|
|
|
2019-08-24 23:08:04 +01:00
|
|
|
if (area_size == _size && area_size == other._size) {
|
|
|
|
// Equivalent of full copy between two blocks of same size
|
2017-01-01 04:40:16 +01:00
|
|
|
copy_from(other, channel_index);
|
2019-08-24 23:08:04 +01:00
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
} else {
|
2017-01-01 04:40:16 +01:00
|
|
|
if (other_channel.data) {
|
2020-01-04 19:20:49 +00:00
|
|
|
|
2020-07-25 23:19:08 +01:00
|
|
|
if (channel.data == nullptr) {
|
2019-04-22 20:15:19 +01:00
|
|
|
create_channel(channel_index, _size, channel.defval);
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
2020-01-04 19:20:49 +00:00
|
|
|
|
|
|
|
if (channel.depth == DEPTH_8_BIT) {
|
|
|
|
// Native format
|
|
|
|
// Copy row by row
|
|
|
|
Vector3i pos;
|
|
|
|
for (pos.z = 0; pos.z < area_size.z; ++pos.z) {
|
|
|
|
for (pos.x = 0; pos.x < area_size.x; ++pos.x) {
|
|
|
|
// Row direction is Y
|
|
|
|
unsigned int src_ri = other.index(pos.x + src_min.x, pos.y + src_min.y, pos.z + src_min.z);
|
|
|
|
unsigned int dst_ri = index(pos.x + dst_min.x, pos.y + dst_min.y, pos.z + dst_min.z);
|
|
|
|
memcpy(&channel.data[dst_ri], &other_channel.data[src_ri], area_size.y * sizeof(uint8_t));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// TODO Optimized versions
|
|
|
|
Vector3i pos;
|
|
|
|
for (pos.z = 0; pos.z < area_size.z; ++pos.z) {
|
|
|
|
for (pos.x = 0; pos.x < area_size.x; ++pos.x) {
|
|
|
|
for (pos.y = 0; pos.y < area_size.y; ++pos.y) {
|
|
|
|
uint64_t v = other.get_voxel(src_min + pos, channel_index);
|
|
|
|
set_voxel(v, dst_min + pos, channel_index);
|
|
|
|
}
|
|
|
|
}
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
}
|
2020-01-04 19:20:49 +00:00
|
|
|
|
2017-08-13 01:19:39 +02:00
|
|
|
} else if (channel.defval != other_channel.defval) {
|
2020-07-25 23:19:08 +01:00
|
|
|
if (channel.data == nullptr) {
|
2019-04-22 20:15:19 +01:00
|
|
|
create_channel(channel_index, _size, channel.defval);
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
2020-01-04 19:20:49 +00:00
|
|
|
fill_area(other_channel.defval, dst_min, dst_min + area_size, channel_index);
|
2017-01-01 04:40:16 +01:00
|
|
|
}
|
|
|
|
}
|
2016-05-05 21:08:52 +02:00
|
|
|
}
|
|
|
|
|
2019-05-28 00:40:09 +01:00
|
|
|
Ref<VoxelBuffer> VoxelBuffer::duplicate() const {
|
|
|
|
VoxelBuffer *d = memnew(VoxelBuffer);
|
2019-09-06 23:24:56 +01:00
|
|
|
d->create(_size);
|
2019-05-28 00:40:09 +01:00
|
|
|
d->copy_from(*this);
|
|
|
|
return Ref<VoxelBuffer>(d);
|
|
|
|
}
|
|
|
|
|
2020-01-25 23:26:55 +00:00
|
|
|
bool VoxelBuffer::get_channel_raw(unsigned int channel_index, ArraySlice<uint8_t> &slice) const {
|
2018-09-30 01:55:09 +01:00
|
|
|
const Channel &channel = _channels[channel_index];
|
2020-01-25 23:26:55 +00:00
|
|
|
if (channel.data != nullptr) {
|
|
|
|
slice = ArraySlice<uint8_t>(channel.data, 0, channel.size_in_bytes);
|
|
|
|
return true;
|
2020-01-04 19:20:49 +00:00
|
|
|
}
|
2020-01-25 23:26:55 +00:00
|
|
|
slice = ArraySlice<uint8_t>();
|
|
|
|
return false;
|
2018-09-30 01:55:09 +01:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::create_channel(int i, Vector3i size, uint64_t defval) {
|
2017-01-01 04:40:16 +01:00
|
|
|
create_channel_noinit(i, size);
|
2020-01-04 19:20:49 +00:00
|
|
|
fill(defval, i);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t VoxelBuffer::get_size_in_bytes_for_volume(Vector3i size, Depth depth) {
|
|
|
|
// Calculate appropriate size based on bit depth
|
|
|
|
const unsigned int volume = size.x * size.y * size.z;
|
|
|
|
const unsigned int bits = volume * ::get_depth_bit_count(depth);
|
|
|
|
unsigned int size_in_bytes = (bits >> 3);
|
|
|
|
return size_in_bytes;
|
2016-05-05 21:08:52 +02:00
|
|
|
}
|
2016-05-01 15:00:02 +02:00
|
|
|
|
2016-05-05 21:08:52 +02:00
|
|
|
void VoxelBuffer::create_channel_noinit(int i, Vector3i size) {
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[i];
|
2020-01-04 19:20:49 +00:00
|
|
|
uint32_t size_in_bytes = get_size_in_bytes_for_volume(size, channel.depth);
|
2020-01-26 01:59:53 +00:00
|
|
|
CRASH_COND(channel.data != nullptr);
|
2020-01-04 19:20:49 +00:00
|
|
|
channel.data = allocate_channel_data(size_in_bytes);
|
|
|
|
channel.size_in_bytes = size_in_bytes;
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2017-01-05 02:39:40 +01:00
|
|
|
void VoxelBuffer::delete_channel(int i) {
|
2017-08-13 01:19:39 +02:00
|
|
|
Channel &channel = _channels[i];
|
2020-01-02 20:31:05 +00:00
|
|
|
ERR_FAIL_COND(channel.data == nullptr);
|
2020-01-04 19:20:49 +00:00
|
|
|
free_channel_data(channel.data, channel.size_in_bytes);
|
2020-01-02 20:31:05 +00:00
|
|
|
channel.data = nullptr;
|
2020-01-04 19:20:49 +00:00
|
|
|
channel.size_in_bytes = 0;
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2019-09-02 21:14:30 +01:00
|
|
|
void VoxelBuffer::downscale_to(VoxelBuffer &dst, Vector3i src_min, Vector3i src_max, Vector3i dst_min) const {
|
|
|
|
|
|
|
|
// TODO Align input to multiple of two
|
|
|
|
|
|
|
|
src_min.clamp_to(Vector3i(), _size);
|
2019-09-06 23:24:56 +01:00
|
|
|
src_max.clamp_to(Vector3i(), _size + Vector3i(1));
|
2019-09-02 21:14:30 +01:00
|
|
|
|
2019-09-06 23:24:56 +01:00
|
|
|
Vector3i dst_max = dst_min + ((src_max - src_min) >> 1);
|
2019-09-02 21:14:30 +01:00
|
|
|
|
|
|
|
dst_min.clamp_to(Vector3i(), dst._size);
|
2019-09-06 23:24:56 +01:00
|
|
|
dst_max.clamp_to(Vector3i(), dst._size + Vector3i(1));
|
2019-09-02 21:14:30 +01:00
|
|
|
|
|
|
|
for (int channel_index = 0; channel_index < MAX_CHANNELS; ++channel_index) {
|
|
|
|
|
|
|
|
const Channel &src_channel = _channels[channel_index];
|
2019-09-06 23:24:56 +01:00
|
|
|
const Channel &dst_channel = dst._channels[channel_index];
|
2019-09-02 21:14:30 +01:00
|
|
|
|
|
|
|
if (src_channel.data == nullptr && dst_channel.data == nullptr && src_channel.defval == dst_channel.defval) {
|
|
|
|
// No action needed
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Nearest-neighbor downscaling
|
|
|
|
|
|
|
|
Vector3i pos;
|
|
|
|
for (pos.z = dst_min.z; pos.z < dst_max.z; ++pos.z) {
|
|
|
|
for (pos.x = dst_min.x; pos.x < dst_max.x; ++pos.x) {
|
|
|
|
for (pos.y = dst_min.y; pos.y < dst_max.y; ++pos.y) {
|
|
|
|
|
2020-02-05 21:08:23 +01:00
|
|
|
const Vector3i src_pos = src_min + ((pos - dst_min) << 1);
|
2019-09-02 21:14:30 +01:00
|
|
|
|
|
|
|
// TODO Remove check once it works
|
|
|
|
CRASH_COND(!validate_pos(src_pos.x, src_pos.y, src_pos.z));
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
uint64_t v;
|
2019-09-02 21:14:30 +01:00
|
|
|
if (src_channel.data) {
|
2020-01-04 19:20:49 +00:00
|
|
|
// TODO Optimized version?
|
2020-02-05 21:25:11 +00:00
|
|
|
v = get_voxel(src_pos, channel_index);
|
2019-09-02 21:14:30 +01:00
|
|
|
} else {
|
|
|
|
v = src_channel.defval;
|
|
|
|
}
|
|
|
|
|
|
|
|
dst.set_voxel(v, pos, channel_index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-14 21:34:41 +00:00
|
|
|
Ref<VoxelTool> VoxelBuffer::get_voxel_tool() {
|
|
|
|
// I can't make this function `const`, because `Ref<T>` has no constructor taking a `const T*`.
|
|
|
|
// The compiler would then choose Ref<T>(const Variant&), which fumbles `this` into a null pointer
|
|
|
|
Ref<VoxelBuffer> vb(this);
|
|
|
|
return Ref<VoxelTool>(memnew(VoxelToolBuffer(vb)));
|
2019-09-03 22:54:40 +01:00
|
|
|
}
|
|
|
|
|
2019-09-06 23:24:56 +01:00
|
|
|
bool VoxelBuffer::equals(const VoxelBuffer *p_other) const {
|
|
|
|
CRASH_COND(p_other == nullptr);
|
|
|
|
|
|
|
|
if (p_other->_size != _size) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int channel_index = 0; channel_index < MAX_CHANNELS; ++channel_index) {
|
|
|
|
|
|
|
|
const Channel &channel = _channels[channel_index];
|
|
|
|
const Channel &other_channel = p_other->_channels[channel_index];
|
|
|
|
|
|
|
|
if ((channel.data == nullptr) != (other_channel.data == nullptr)) {
|
|
|
|
// Note: they could still logically be equal if one channel contains uniform voxel memory
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
if (channel.depth != other_channel.depth) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-09-06 23:24:56 +01:00
|
|
|
if (channel.data == nullptr) {
|
|
|
|
if (channel.defval != other_channel.defval) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
2020-01-04 19:20:49 +00:00
|
|
|
CRASH_COND(channel.size_in_bytes != other_channel.size_in_bytes);
|
|
|
|
for (unsigned int i = 0; i < channel.size_in_bytes; ++i) {
|
2019-09-06 23:24:56 +01:00
|
|
|
if (channel.data[i] != other_channel.data[i]) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::set_channel_depth(unsigned int channel_index, Depth new_depth) {
|
|
|
|
ERR_FAIL_INDEX(channel_index, MAX_CHANNELS);
|
|
|
|
ERR_FAIL_INDEX(new_depth, DEPTH_COUNT);
|
|
|
|
Channel &channel = _channels[channel_index];
|
|
|
|
if (channel.depth == new_depth) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (channel.data != nullptr) {
|
|
|
|
// TODO Implement conversion
|
|
|
|
WARN_PRINT("Changing VoxelBuffer depth with present data, this will reset the channel");
|
|
|
|
delete_channel(channel_index);
|
|
|
|
}
|
|
|
|
channel.defval = clamp_value_for_depth(channel.defval, new_depth);
|
2020-07-08 19:59:02 +01:00
|
|
|
channel.depth = new_depth;
|
2020-01-04 19:20:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VoxelBuffer::Depth VoxelBuffer::get_channel_depth(unsigned int channel_index) const {
|
|
|
|
ERR_FAIL_INDEX_V(channel_index, MAX_CHANNELS, DEPTH_8_BIT);
|
|
|
|
return _channels[channel_index].depth;
|
|
|
|
}
|
|
|
|
|
2020-01-06 00:46:58 +00:00
|
|
|
uint32_t VoxelBuffer::get_depth_bit_count(Depth d) {
|
2020-01-04 19:20:49 +00:00
|
|
|
return ::get_depth_bit_count(d);
|
|
|
|
}
|
|
|
|
|
2019-09-06 23:24:56 +01:00
|
|
|
Ref<Image> VoxelBuffer::debug_print_sdf_to_image_top_down() {
|
|
|
|
Image *im = memnew(Image);
|
|
|
|
im->create(_size.x, _size.z, false, Image::FORMAT_RGB8);
|
|
|
|
im->lock();
|
|
|
|
Vector3i pos;
|
|
|
|
for (pos.z = 0; pos.z < _size.z; ++pos.z) {
|
|
|
|
for (pos.x = 0; pos.x < _size.x; ++pos.x) {
|
|
|
|
for (pos.y = _size.y - 1; pos.y >= 0; --pos.y) {
|
2020-01-02 21:03:44 +00:00
|
|
|
float v = get_voxel_f(pos.x, pos.y, pos.z, CHANNEL_SDF);
|
2019-09-06 23:24:56 +01:00
|
|
|
if (v < 0.0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
float h = pos.y;
|
|
|
|
float c = h / _size.y;
|
|
|
|
im->set_pixel(pos.x, pos.z, Color(c, c, c));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
im->unlock();
|
|
|
|
return Ref<Image>(im);
|
|
|
|
}
|
|
|
|
|
2016-05-01 15:00:02 +02:00
|
|
|
void VoxelBuffer::_bind_methods() {
|
2019-09-06 23:24:56 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("create", "sx", "sy", "sz"), &VoxelBuffer::_b_create);
|
2017-03-25 01:23:36 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("clear"), &VoxelBuffer::clear);
|
2016-05-01 15:00:02 +02:00
|
|
|
|
2019-09-06 23:24:56 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("get_size"), &VoxelBuffer::_b_get_size);
|
2017-03-25 01:23:36 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("get_size_x"), &VoxelBuffer::get_size_x);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_size_y"), &VoxelBuffer::get_size_y);
|
|
|
|
ClassDB::bind_method(D_METHOD("get_size_z"), &VoxelBuffer::get_size_z);
|
2016-05-01 15:00:02 +02:00
|
|
|
|
2019-09-06 23:24:56 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("set_voxel", "value", "x", "y", "z", "channel"), &VoxelBuffer::_b_set_voxel, DEFVAL(0));
|
|
|
|
ClassDB::bind_method(D_METHOD("set_voxel_f", "value", "x", "y", "z", "channel"), &VoxelBuffer::_b_set_voxel_f, DEFVAL(0));
|
|
|
|
ClassDB::bind_method(D_METHOD("set_voxel_v", "value", "pos", "channel"), &VoxelBuffer::_b_set_voxel_v, DEFVAL(0));
|
|
|
|
ClassDB::bind_method(D_METHOD("get_voxel", "x", "y", "z", "channel"), &VoxelBuffer::_b_get_voxel, DEFVAL(0));
|
2019-04-28 16:28:49 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("get_voxel_f", "x", "y", "z", "channel"), &VoxelBuffer::get_voxel_f, DEFVAL(0));
|
2019-12-14 21:34:41 +00:00
|
|
|
ClassDB::bind_method(D_METHOD("get_voxel_tool"), &VoxelBuffer::get_voxel_tool);
|
2016-05-01 15:00:02 +02:00
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
ClassDB::bind_method(D_METHOD("get_channel_depth", "channel"), &VoxelBuffer::get_channel_depth);
|
|
|
|
ClassDB::bind_method(D_METHOD("set_channel_depth", "channel", "depth"), &VoxelBuffer::set_channel_depth);
|
|
|
|
|
2017-03-25 01:23:36 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("fill", "value", "channel"), &VoxelBuffer::fill, DEFVAL(0));
|
2019-04-28 16:28:49 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("fill_f", "value", "channel"), &VoxelBuffer::fill_f, DEFVAL(0));
|
2019-09-06 23:24:56 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("fill_area", "value", "min", "max", "channel"), &VoxelBuffer::_b_fill_area, DEFVAL(0));
|
2020-01-04 19:20:49 +00:00
|
|
|
ClassDB::bind_method(D_METHOD("copy_channel_from", "other", "channel"), &VoxelBuffer::_b_copy_channel_from);
|
|
|
|
ClassDB::bind_method(D_METHOD("copy_channel_from_area", "other", "src_min", "src_max", "dst_min", "channel"), &VoxelBuffer::_b_copy_channel_from_area);
|
2019-12-26 20:30:20 +00:00
|
|
|
ClassDB::bind_method(D_METHOD("downscale_to", "dst", "src_min", "src_max", "dst_min"), &VoxelBuffer::_b_downscale_to);
|
2016-05-01 15:00:02 +02:00
|
|
|
|
2019-04-28 00:27:17 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("is_uniform", "channel"), &VoxelBuffer::is_uniform);
|
2019-05-25 15:51:59 +01:00
|
|
|
ClassDB::bind_method(D_METHOD("optimize"), &VoxelBuffer::compress_uniform_channels);
|
2019-04-21 19:31:35 +01:00
|
|
|
|
|
|
|
BIND_ENUM_CONSTANT(CHANNEL_TYPE);
|
2020-01-02 21:03:44 +00:00
|
|
|
BIND_ENUM_CONSTANT(CHANNEL_SDF);
|
2019-04-21 19:31:35 +01:00
|
|
|
BIND_ENUM_CONSTANT(CHANNEL_DATA2);
|
|
|
|
BIND_ENUM_CONSTANT(CHANNEL_DATA3);
|
2019-04-22 21:17:19 +01:00
|
|
|
BIND_ENUM_CONSTANT(CHANNEL_DATA4);
|
|
|
|
BIND_ENUM_CONSTANT(CHANNEL_DATA5);
|
|
|
|
BIND_ENUM_CONSTANT(CHANNEL_DATA6);
|
|
|
|
BIND_ENUM_CONSTANT(CHANNEL_DATA7);
|
2019-04-21 19:31:35 +01:00
|
|
|
BIND_ENUM_CONSTANT(MAX_CHANNELS);
|
2020-01-04 19:20:49 +00:00
|
|
|
|
|
|
|
BIND_ENUM_CONSTANT(DEPTH_8_BIT);
|
|
|
|
BIND_ENUM_CONSTANT(DEPTH_16_BIT);
|
|
|
|
BIND_ENUM_CONSTANT(DEPTH_32_BIT);
|
|
|
|
BIND_ENUM_CONSTANT(DEPTH_64_BIT);
|
|
|
|
BIND_ENUM_CONSTANT(DEPTH_COUNT);
|
2016-05-01 15:00:02 +02:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::_b_copy_channel_from(Ref<VoxelBuffer> other, unsigned int channel) {
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_COND(other.is_null());
|
|
|
|
copy_from(**other, channel);
|
2016-05-05 21:08:52 +02:00
|
|
|
}
|
|
|
|
|
2020-01-04 19:20:49 +00:00
|
|
|
void VoxelBuffer::_b_copy_channel_from_area(Ref<VoxelBuffer> other, Vector3 src_min, Vector3 src_max, Vector3 dst_min, unsigned int channel) {
|
2017-01-01 04:40:16 +01:00
|
|
|
ERR_FAIL_COND(other.is_null());
|
|
|
|
copy_from(**other, Vector3i(src_min), Vector3i(src_max), Vector3i(dst_min), channel);
|
2016-05-05 21:08:52 +02:00
|
|
|
}
|
2019-12-26 20:30:20 +00:00
|
|
|
|
|
|
|
void VoxelBuffer::_b_downscale_to(Ref<VoxelBuffer> dst, Vector3 src_min, Vector3 src_max, Vector3 dst_min) const {
|
|
|
|
ERR_FAIL_COND(dst.is_null());
|
|
|
|
downscale_to(**dst, Vector3i(src_min), Vector3i(src_max), Vector3i(dst_min));
|
|
|
|
}
|