I've got an N dimensional (regular) grid with data, which doesn't fill it's volume completely, but should be convex. So, for example, in 2D this is okay (1=exists, 0=missing):
0011111100 0111111110 1111111111 0011111100 0000011100
But this is not:
0011111100 0111101110 1111111111 0011111000 0000011100
I want to find the additional zeros in this second example (marked in bold). And I want to do this in more than 2 dimensions.
The only way I can think of now is to get all possible coordinates in N-1 dimensions and check in the Nth dimension, whether it's convex, which just means finding the first and last data points in that dimension and check, whether any point is missing in between. But I'd have to do that in every dimension and for every slice in that dimension.
There must be an easier solution, right?
To easily solve this problem, consider an opposite problem: "How can one find the space surrounding a shape?"
If you know how to fill in all the space surrounding a shape, then all pixels or voxels which are not part of that empty space must be part of the convex hull. In the following example, we:
0
space with-
-
with0
and0
with1
(results in the convex hull)In the final result, a
1
now represents enclosed empty space. Note that-
could be represented as a2
or3
for this to work and we would temporarily need two bits per voxel, not just one.To implement the first step, we simply flood-fill all
0
voxels with-
for all voxels that are on the edge of the model. For each of those boundary voxels we start a separate flood-fill run. The empty space inside of the model stays unchanged, because the flood-fill doesn't reach it.Implementing the second and third step should be trivial.
Avoiding "Troolean" Logic
It's also possible to do all of this with only boolean logic, so no third
-
value. To do this, we would have to:1
instead of-
1
into a separate output bufferThe first step would then give us a result of:
We now just need to bit-flip this buffer to get the convex hull.