Flywheel/joml/sampling/BestCandidateSampling.java
PepperCode1 dd18300b70 Scheme-a-version
- Fix Resources not being closed properly
- Change versioning scheme to match Create
- Add LICENSE to built jar
- Fix mods.toml version sync
- Move JOML code to non-src directory
- Update Gradle
- Organize imports
2022-07-15 00:00:54 -07:00

1034 lines
40 KiB
Java

/*
* The MIT License
*
* Copyright (c) 2016-2021 JOML
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package com.jozufozu.flywheel.repack.joml.sampling;
import java.nio.FloatBuffer;
import java.util.ArrayList;
import com.jozufozu.flywheel.repack.joml.Random;
import com.jozufozu.flywheel.repack.joml.Vector2f;
import com.jozufozu.flywheel.repack.joml.Vector3f;
/**
* Creates samples using the "Best Candidate" algorithm.
*
* @author Kai Burjack
*/
public class BestCandidateSampling {
private static final class IntHolder {
int value;
}
/**
* Generates Best Candidate samples on a unit sphere.
* <p>
* References:
* <ul>
* <li><a href="https://arxiv.org/ftp/cs/papers/0701/0701164.pdf">Indexing the Sphere with the Hierarchical Triangular Mesh</a>
* <li><a href="http://math.stackexchange.com/questions/1244512/point-in-a-spherical-triangle-test">Point in a spherical triangle test</a>
* </ul>
*
* @author Kai Burjack
*/
public static class Sphere {
/**
* Implementation of a Hierarchical Triangular Mesh structure to index the sample points on the unit sphere for accelerating 1-nearest neighbor searches.
*
* @author Kai Burjack
*/
private static final class Node {
private static final int MAX_OBJECTS_PER_NODE = 32;
private float v0x, v0y, v0z;
private float v1x, v1y, v1z;
private float v2x, v2y, v2z;
private float cx, cy, cz;
private float arc;
private ArrayList objects;
private Node[] children;
Node() {
this.children = new Node[8];
float s = 1f;
this.arc = 2.0f * (float) Math.PI;
/*
* See: https://arxiv.org/ftp/cs/papers/0701/0701164.pdf
*/
this.children[0] = new Node(-s, 0, 0, 0, 0, s, 0, s, 0);
this.children[1] = new Node(0, 0, s, s, 0, 0, 0, s, 0);
this.children[2] = new Node(s, 0, 0, 0, 0, -s, 0, s, 0);
this.children[3] = new Node(0, 0, -s, -s, 0, 0, 0, s, 0);
this.children[4] = new Node(-s, 0, 0, 0, -s, 0, 0, 0, s);
this.children[5] = new Node(0, 0, s, 0, -s, 0, s, 0, 0);
this.children[6] = new Node(s, 0, 0, 0, -s, 0, 0, 0, -s);
this.children[7] = new Node(0, 0, -s, 0, -s, 0, -s, 0, 0);
}
private Node(float x0, float y0, float z0, float x1, float y1, float z1, float x2, float y2, float z2) {
this.v0x = x0;
this.v0y = y0;
this.v0z = z0;
this.v1x = x1;
this.v1y = y1;
this.v1z = z1;
this.v2x = x2;
this.v2y = y2;
this.v2z = z2;
cx = (v0x + v1x + v2x) / 3.0f;
cy = (v0y + v1y + v2y) / 3.0f;
cz = (v0z + v1z + v2z) / 3.0f;
float invCLen = Math.invsqrt(cx * cx + cy * cy + cz * cz);
cx *= invCLen;
cy *= invCLen;
cz *= invCLen;
// Compute maximum radius around triangle centroid
float arc1 = greatCircleDist(cx, cy, cz, v0x, v0y, v0z);
float arc2 = greatCircleDist(cx, cy, cz, v1x, v1y, v1z);
float arc3 = greatCircleDist(cx, cy, cz, v2x, v2y, v2z);
float dist = Math.max(Math.max(arc1, arc2), arc3);
/*
* Convert radius into diameter, but also take into account the linear
* arccos approximation we use.
* This value was obtained empirically!
*/
dist *= 1.7f;
this.arc = dist;
}
private void split() {
float w0x = v1x + v2x;
float w0y = v1y + v2y;
float w0z = v1z + v2z;
float len0 = Math.invsqrt(w0x * w0x + w0y * w0y + w0z * w0z);
w0x *= len0;
w0y *= len0;
w0z *= len0;
float w1x = v0x + v2x;
float w1y = v0y + v2y;
float w1z = v0z + v2z;
float len1 = Math.invsqrt(w1x * w1x + w1y * w1y + w1z * w1z);
w1x *= len1;
w1y *= len1;
w1z *= len1;
float w2x = v0x + v1x;
float w2y = v0y + v1y;
float w2z = v0z + v1z;
float len2 = Math.invsqrt(w2x * w2x + w2y * w2y + w2z * w2z);
w2x *= len2;
w2y *= len2;
w2z *= len2;
children = new Node[4];
/*
* See: https://arxiv.org/ftp/cs/papers/0701/0701164.pdf
*/
children[0] = new Node(v0x, v0y, v0z, w2x, w2y, w2z, w1x, w1y, w1z);
children[1] = new Node(v1x, v1y, v1z, w0x, w0y, w0z, w2x, w2y, w2z);
children[2] = new Node(v2x, v2y, v2z, w1x, w1y, w1z, w0x, w0y, w0z);
children[3] = new Node(w0x, w0y, w0z, w1x, w1y, w1z, w2x, w2y, w2z);
}
private void insertIntoChild(Vector3f o) {
for (int i = 0; i < children.length; i++) {
Node c = children[i];
/*
* Idea: Test whether ray from origin towards point cuts triangle
*
* See: http://math.stackexchange.com/questions/1244512/point-in-a-spherical-triangle-test
*/
if (isPointOnSphericalTriangle(o.x, o.y, o.z, c.v0x, c.v0y, c.v0z, c.v1x, c.v1y, c.v1z, c.v2x, c.v2y, c.v2z, 1E-6f)) {
c.insert(o);
return;
}
}
}
void insert(Vector3f object) {
if (children != null) {
insertIntoChild(object);
return;
}
if (objects != null && objects.size() == MAX_OBJECTS_PER_NODE) {
split();
for (int i = 0; i < MAX_OBJECTS_PER_NODE; i++)
insertIntoChild((Vector3f) objects.get(i));
objects = null;
insertIntoChild(object);
} else {
if (objects == null)
objects = new ArrayList(MAX_OBJECTS_PER_NODE);
objects.add(object);
}
}
/**
* This is essentially a ray cast from the origin of the sphere to the point to test and then checking whether that ray goes through the triangle.
* <p>
* Reference: <a href="http://www.graphics.cornell.edu/pubs/1997/MT97.pdf">Fast,
* Minimum Storage Ray/Triangle Intersection</a>
*/
private static boolean isPointOnSphericalTriangle(float x, float y, float z, float v0X, float v0Y, float v0Z, float v1X, float v1Y, float v1Z, float v2X, float v2Y,
float v2Z, float epsilon) {
float edge1X = v1X - v0X;
float edge1Y = v1Y - v0Y;
float edge1Z = v1Z - v0Z;
float edge2X = v2X - v0X;
float edge2Y = v2Y - v0Y;
float edge2Z = v2Z - v0Z;
float pvecX = y * edge2Z - z * edge2Y;
float pvecY = z * edge2X - x * edge2Z;
float pvecZ = x * edge2Y - y * edge2X;
float det = edge1X * pvecX + edge1Y * pvecY + edge1Z * pvecZ;
if (det > -epsilon && det < epsilon)
return false;
float tvecX = -v0X;
float tvecY = -v0Y;
float tvecZ = -v0Z;
float invDet = 1.0f / det;
float u = (tvecX * pvecX + tvecY * pvecY + tvecZ * pvecZ) * invDet;
if (u < 0.0f || u > 1.0f)
return false;
float qvecX = tvecY * edge1Z - tvecZ * edge1Y;
float qvecY = tvecZ * edge1X - tvecX * edge1Z;
float qvecZ = tvecX * edge1Y - tvecY * edge1X;
float v = (x * qvecX + y * qvecY + z * qvecZ) * invDet;
if (v < 0.0f || u + v > 1.0f)
return false;
float t = (edge2X * qvecX + edge2Y * qvecY + edge2Z * qvecZ) * invDet;
return t >= epsilon;
}
private int child(float x, float y, float z) {
for (int i = 0; i < children.length; i++) {
Node c = children[i];
if (isPointOnSphericalTriangle(x, y, z, c.v0x, c.v0y, c.v0z, c.v1x, c.v1y, c.v1z, c.v2x, c.v2y, c.v2z, 1E-5f))
return i;
}
// No child found. This can happen in 'nearest()' when querying possible nearby nodes
return 0;
}
/**
* Reference: <a href="https://en.wikipedia.org/wiki/Great-circle_distance#Vector_version">https://en.wikipedia.org/</a>
*/
private float greatCircleDist(float x1, float y1, float z1, float x2, float y2, float z2) {
float dot = x1 * x2 + y1 * y2 + z1 * z2;
/*
* Just use a linear function, because we (mostly) do less-than comparisons on the result.
* We just need a linear function which:
* f(-1) = PI
* f(0) = PI/2
* f(1) = 0
*/
return (float) (-Math.PIHalf * dot + Math.PIHalf);
//return (float) Math.acos(dot);
}
float nearest(float x, float y, float z) {
return nearest(x, y, z, Float.POSITIVE_INFINITY);
}
float nearest(float x, float y, float z, float n) {
float gcd = greatCircleDist(x, y, z, cx, cy, cz);
/*
* If great-circle-distance between query point and centroid is larger than the current smallest distance 'n' plus the great circle diameter 'arc', we abort here,
* because then it is not possible for any point in the triangle patch to be closer to the query point than 'n'.
*/
/*
* Yes, we are subtracting two great-circle distances from one another here, which we did not even compute correctly
* using our overly linear arccos approximation. But the 1.7 factor above will take care that we still stay conservative
* enough here and not rejecting triangle patches which would contain samples nearer than 'n'.
*/
if (gcd - arc > n)
return n;
float nr = n;
if (children != null) {
int num = children.length, mod = num-1;
for (int i = child(x, y, z), c = 0; c < num; i = (i + 1) & mod, c++) {
float n1 = children[i].nearest(x, y, z, nr);
nr = Math.min(n1, nr);
}
return nr;
}
for (int i = 0; objects != null && i < objects.size(); i++) {
Vector3f o = (Vector3f) objects.get(i);
float d = greatCircleDist(o.x, o.y, o.z, x, y, z);
if (d < nr)
nr = d;
}
return nr;
}
}
private boolean onHemisphere;
private int numSamples;
private int numCandidates = 60; // <- use a reasonable default
private long seed;
/**
* Create a new instance of {@link Sphere} to configure and generate 'best candidate' sample positions on the unit sphere.
*/
public Sphere() {}
/**
* Generate 'best candidate' sample positions and store the coordinates of all generated samples into the given <code>xyzs</code> float array.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param xyzs
* will hold the x, y and z coordinates of all samples in the order <code>XYZXYZXYZ...</code>.
* This array must have a length of at least <code>numSamples</code>
* @return this
*/
public Sphere generate(final float[] xyzs) {
final IntHolder i = new IntHolder();
return generate(new Callback3d() {
public void onNewSample(float x, float y, float z) {
xyzs[3 * i.value + 0] = x;
xyzs[3 * i.value + 1] = y;
xyzs[3 * i.value + 2] = z;
i.value++;
}
});
}
/**
* Generate 'best candidate' sample positions and store the coordinates of all generated samples into the given <code>xyzs</code> FloatBuffer.
* <p>
* The samples will be written starting at the current position of the FloatBuffer. The position of the FloatBuffer will not be modified.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param xyzs
* will hold the x, y and z coordinates of all samples in the order <code>XYZXYZXYZ...</code>.
* This FloatBuffer must have at least <code>numSamples</code> remaining elements.
* The position of the buffer will not be modified by this method
* @return this
*/
public Sphere generate(final FloatBuffer xyzs) {
final IntHolder i = new IntHolder();
final int pos = xyzs.position();
return generate(new Callback3d() {
public void onNewSample(float x, float y, float z) {
xyzs.put(pos + 3 * i.value + 0, x);
xyzs.put(pos + 3 * i.value + 1, y);
xyzs.put(pos + 3 * i.value + 2, z);
i.value++;
}
});
}
/**
* Set the seed to initialize the pseudo-random number generator with.
*
* @param seed
* the seed value
* @return this
*/
public Sphere seed(long seed) {
this.seed = seed;
return this;
}
/**
* Set the number of samples to generate.
*
* @param numSamples
* the number of samples
* @return this
*/
public Sphere numSamples(int numSamples) {
this.numSamples = numSamples;
return this;
}
/**
* Set the number of candidates to try for each generated sample.
*
* @param numCandidates
* the number of candidates to try
* @return this
*/
public Sphere numCandidates(int numCandidates) {
this.numCandidates = numCandidates;
return this;
}
/**
* Set whether to generate samples on a hemisphere around the <code>+Z</code> axis.
* <p>
* The default is <code>false</code>, which will generate samples on the whole unit sphere.
*
* @param onHemisphere
* whether to generate samples on the hemisphere
* @return this
*/
public Sphere onHemisphere(boolean onHemisphere) {
this.onHemisphere = onHemisphere;
return this;
}
/**
* Generate 'best candidate' sample call the given <code>callback</code> for each generated sample.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param callback
* will be called with the coordinates of each generated sample position
* @return this
*/
public Sphere generate(Callback3d callback) {
Random rnd = new Random(seed);
Node otree = new Node();
for (int i = 0; i < numSamples; i++) {
float bestX = Float.NaN, bestY = Float.NaN, bestZ = Float.NaN, bestDist = 0.0f;
for (int c = 0; c < numCandidates; c++) {
/*
* Random point on sphere
*
* Reference: <a href="http://mathworld.wolfram.com/SpherePointPicking.html">http://mathworld.wolfram.com/</a>
*/
float x1, x2;
do {
x1 = rnd.nextFloat() * 2.0f - 1.0f;
x2 = rnd.nextFloat() * 2.0f - 1.0f;
} while (x1 * x1 + x2 * x2 > 1.0f);
float sqrt = (float) Math.sqrt(1.0 - x1 * x1 - x2 * x2);
float x = 2 * x1 * sqrt;
float y = 2 * x2 * sqrt;
float z = 1.0f - 2.0f * (x1 * x1 + x2 * x2);
if (onHemisphere) {
z = Math.abs(z);
}
float minDist = otree.nearest(x, y, z);
if (minDist > bestDist) {
bestDist = minDist;
bestX = x;
bestY = y;
bestZ = z;
}
}
callback.onNewSample(bestX, bestY, bestZ);
otree.insert(new Vector3f(bestX, bestY, bestZ));
}
return this;
}
}
/**
* Simple quatree that can handle points and 1-nearest neighbor search.
*
* @author Kai Burjack
*/
private static class QuadTree {
private static final int MAX_OBJECTS_PER_NODE = 32;
// Constants for the quadrants of the quadtree
private static final int PXNY = 0;
private static final int NXNY = 1;
private static final int NXPY = 2;
private static final int PXPY = 3;
private float minX, minY, hs;
private ArrayList objects;
private QuadTree[] children;
QuadTree(float minX, float minY, float size) {
this.minX = minX;
this.minY = minY;
this.hs = size * 0.5f;
}
private void split() {
children = new QuadTree[4];
children[NXNY] = new QuadTree(minX, minY, hs);
children[PXNY] = new QuadTree(minX + hs, minY, hs);
children[NXPY] = new QuadTree(minX, minY + hs, hs);
children[PXPY] = new QuadTree(minX + hs, minY + hs, hs);
}
private void insertIntoChild(Vector2f o) {
children[quadrant(o.x, o.y)].insert(o);
}
void insert(Vector2f object) {
if (children != null) {
insertIntoChild(object);
return;
}
if (objects != null && objects.size() == MAX_OBJECTS_PER_NODE) {
split();
for (int i = 0; i < objects.size(); i++)
insertIntoChild((Vector2f) objects.get(i));
objects = null;
insertIntoChild(object);
} else {
if (objects == null)
objects = new ArrayList(MAX_OBJECTS_PER_NODE);
objects.add(object);
}
}
private int quadrant(float x, float y) {
if (x < minX + hs) {
if (y < minY + hs)
return NXNY;
return NXPY;
}
if (y < minY + hs)
return PXNY;
return PXPY;
}
float nearest(float x, float y, float lowerBound, float upperBound) {
float ub = upperBound;
if (x < minX - upperBound || x > minX + hs * 2 + upperBound || y < minY - upperBound
|| y > minY + hs * 2 + upperBound)
return ub;
if (children != null) {
for (int i = quadrant(x, y), c = 0; c < 4; i = (i + 1) & 3, c++) {
float n1 = children[i].nearest(x, y, lowerBound, ub);
ub = Math.min(n1, ub);
if (ub <= lowerBound)
return lowerBound;
}
return ub;
}
float ub2 = ub * ub;
float lb2 = lowerBound * lowerBound;
for (int i = 0; objects != null && i < objects.size(); i++) {
Vector2f o = (Vector2f) objects.get(i);
float d = o.distanceSquared(x, y);
if (d <= lb2)
return lowerBound;
if (d < ub2)
ub2 = d;
}
return (float) Math.sqrt(ub2);
}
}
/**
* Generates Best Candidate samples on a unit disk.
*
* @author Kai Burjack
*/
public static class Disk {
private int numSamples;
private int numCandidates = 60; // <- use a reasonable default
private long seed;
/**
* Create a new instance of {@link Disk} to configure and generate 'best candidate' sample positions on the unit disk.
*/
public Disk() {}
/**
* Set the seed to initialize the pseudo-random number generator with.
*
* @param seed
* the seed value
* @return this
*/
public Disk seed(long seed) {
this.seed = seed;
return this;
}
/**
* Set the number of samples to generate.
*
* @param numSamples
* the number of samples
* @return this
*/
public Disk numSamples(int numSamples) {
this.numSamples = numSamples;
return this;
}
/**
* Set the number of candidates to try for each generated sample.
*
* @param numCandidates
* the number of candidates to try
* @return this
*/
public Disk numCandidates(int numCandidates) {
this.numCandidates = numCandidates;
return this;
}
/**
* Generate 'best candidate' sample positions and store the coordinates of all generated samples into the given <code>xys</code> float array.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param xys
* will hold the x and y coordinates of all samples in the order <code>XYXYXY...</code>.
* This array must have a length of at least <code>numSamples</code>
* @return this
*/
public Disk generate(final float[] xys) {
final IntHolder i = new IntHolder();
return generate(new Callback2d() {
public void onNewSample(float x, float y) {
xys[2 * i.value + 0] = x;
xys[2 * i.value + 1] = y;
i.value++;
}
});
}
/**
* Generate 'best candidate' sample positions and store the coordinates of all generated samples into the given <code>xys</code> FloatBuffer.
* <p>
* The samples will be written starting at the current position of the FloatBuffer. The position of the FloatBuffer will not be modified.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param xys
* will hold the x and y coordinates of all samples in the order <code>XYXYXY...</code>. This FloatBuffer must have at least <code>numSamples</code> remaining elements. The
* position of the buffer will not be modified by this method
* @return this
*/
public Disk generate(final FloatBuffer xys) {
final IntHolder i = new IntHolder();
final int pos = xys.position();
return generate(new Callback2d() {
public void onNewSample(float x, float y) {
xys.put(pos + 3 * i.value + 0, x);
xys.put(pos + 3 * i.value + 1, y);
i.value++;
}
});
}
/**
* Generate 'best candidate' sample positions and call the given <code>callback</code> for each generated sample.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param callback
* will be called with the coordinates of each generated sample position
* @return this
*/
public Disk generate(Callback2d callback) {
QuadTree qtree = new QuadTree(-1, -1, 2);
Random rnd = new Random(seed);
for (int i = 0; i < numSamples; i++) {
float bestX = 0, bestY = 0, bestDist = 0.0f;
for (int c = 0; c < numCandidates; c++) {
float x, y;
do {
x = rnd.nextFloat() * 2.0f - 1.0f;
y = rnd.nextFloat() * 2.0f - 1.0f;
} while (x * x + y * y > 1.0f);
float minDist = qtree.nearest(x, y, bestDist, Float.POSITIVE_INFINITY);
if (minDist > bestDist) {
bestDist = minDist;
bestX = x;
bestY = y;
}
}
callback.onNewSample(bestX, bestY);
qtree.insert(new Vector2f(bestX, bestY));
}
return this;
}
}
/**
* Generates Best Candidate samples on a unit quad.
*
* @author Kai Burjack
*/
public static class Quad {
private int numSamples;
private int numCandidates = 60; // <- use a reasonable default
private long seed;
/**
* Create a new instance of {@link Quad} to configure and generate 'best candidate' sample positions on the unit quad.
*/
public Quad() {}
/**
* Set the seed to initialize the pseudo-random number generator with.
*
* @param seed
* the seed value
* @return this
*/
public Quad seed(long seed) {
this.seed = seed;
return this;
}
/**
* Set the number of samples to generate.
*
* @param numSamples
* the number of samples
* @return this
*/
public Quad numSamples(int numSamples) {
this.numSamples = numSamples;
return this;
}
/**
* Set the number of candidates to try for each generated sample.
*
* @param numCandidates
* the number of candidates to try
* @return this
*/
public Quad numCandidates(int numCandidates) {
this.numCandidates = numCandidates;
return this;
}
/**
* Generate 'best candidate' sample positions and store the coordinates of all generated samples into the given <code>xyzs</code> float array.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param xyzs
* will hold the x, y and z coordinates of all samples in the order <code>XYZXYZXYZ...</code>.
* This array must have a length of at least <code>numSamples</code>
* @return this
*/
public Quad generate(final float[] xyzs) {
final IntHolder i = new IntHolder();
return generate(new Callback2d() {
public void onNewSample(float x, float y) {
xyzs[2 * i.value + 0] = x;
xyzs[2 * i.value + 1] = y;
i.value++;
}
});
}
/**
* Generate 'best candidate' sample positions and store the coordinates of all generated samples into the given <code>xys</code> FloatBuffer.
* <p>
* The samples will be written starting at the current position of the FloatBuffer. The position of the FloatBuffer will not be modified.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param xys
* will hold the x and y coordinates of all samples in the order <code>XYXYXY...</code>. This FloatBuffer must have at least <code>numSamples</code> remaining elements. The position of
* the buffer will not be modified by this method
* @return this
*/
public Quad generate(final FloatBuffer xys) {
final IntHolder i = new IntHolder();
final int pos = xys.position();
return generate(new Callback2d() {
public void onNewSample(float x, float y) {
xys.put(pos + 3 * i.value + 0, x);
xys.put(pos + 3 * i.value + 1, y);
i.value++;
}
});
}
/**
* Generate 'best candidate' sample positions and call the given <code>callback</code> for each generated sample.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param callback
* will be called with the coordinates of each generated sample position
* @return this
*/
public Quad generate(Callback2d callback) {
QuadTree qtree = new QuadTree(-1, -1, 2);
Random rnd = new Random(seed);
for (int i = 0; i < numSamples; i++) {
float bestX = 0, bestY = 0, bestDist = 0.0f;
for (int c = 0; c < numCandidates; c++) {
float x = rnd.nextFloat() * 2.0f - 1.0f;
float y = rnd.nextFloat() * 2.0f - 1.0f;
float minDist = qtree.nearest(x, y, bestDist, Float.POSITIVE_INFINITY);
if (minDist > bestDist) {
bestDist = minDist;
bestX = x;
bestY = y;
}
}
callback.onNewSample(bestX, bestY);
qtree.insert(new Vector2f(bestX, bestY));
}
return this;
}
}
/**
* Simple octree for points and 1-nearest neighbor distance query.
*
* @author Kai Burjack
*/
private static class Octree {
private static final int MAX_OBJECTS_PER_NODE = 32;
// Constants for the octants of the octree
private static final int PXNYNZ = 0;
private static final int NXNYNZ = 1;
private static final int NXPYNZ = 2;
private static final int PXPYNZ = 3;
private static final int PXNYPZ = 4;
private static final int NXNYPZ = 5;
private static final int NXPYPZ = 6;
private static final int PXPYPZ = 7;
private float minX, minY, minZ, hs;
private ArrayList objects;
private Octree[] children;
Octree(float minX, float minY, float minZ, float size) {
this.minX = minX;
this.minY = minY;
this.minZ = minZ;
this.hs = size * 0.5f;
}
private void split() {
children = new Octree[8];
children[NXNYNZ] = new Octree(minX, minY, minZ, hs);
children[PXNYNZ] = new Octree(minX + hs, minY, minZ, hs);
children[NXPYNZ] = new Octree(minX, minY + hs, minZ, hs);
children[PXPYNZ] = new Octree(minX + hs, minY + hs, minZ, hs);
children[NXNYPZ] = new Octree(minX, minY, minZ + hs, hs);
children[PXNYPZ] = new Octree(minX + hs, minY, minZ + hs, hs);
children[NXPYPZ] = new Octree(minX, minY + hs, minZ + hs, hs);
children[PXPYPZ] = new Octree(minX + hs, minY + hs, minZ + hs, hs);
}
private void insertIntoChild(Vector3f o) {
children[octant(o.x, o.y, o.z)].insert(o);
}
void insert(Vector3f object) {
if (children != null) {
insertIntoChild(object);
return;
}
if (objects != null && objects.size() == MAX_OBJECTS_PER_NODE) {
split();
for (int i = 0; i < objects.size(); i++)
insertIntoChild((Vector3f) objects.get(i));
objects = null;
insertIntoChild(object);
} else {
if (objects == null)
objects = new ArrayList(MAX_OBJECTS_PER_NODE);
objects.add(object);
}
}
private int octant(float x, float y, float z) {
if (x < minX + hs)
if (y < minY + hs) {
if (z < minZ + hs)
return NXNYNZ;
return NXNYPZ;
} else if (z < minZ + hs)
return NXPYNZ;
else
return NXPYPZ;
else if (y < minY + hs) {
if (z < minZ + hs)
return PXNYNZ;
return PXNYPZ;
} else if (z < minZ + hs)
return PXPYNZ;
else
return PXPYPZ;
}
float nearest(float x, float y, float z, float lowerBound, float upperBound) {
float up = upperBound;
if (x < minX - upperBound || x > minX + hs * 2 + upperBound || y < minY - upperBound || y > minY + hs * 2 + upperBound ||
z < minZ - upperBound || z > minZ + hs * 2 + upperBound)
return up;
if (children != null) {
for (int i = octant(x, y, z), c = 0; c < 8; i = (i + 1) & 7, c++) {
float n1 = children[i].nearest(x, y, z, lowerBound, up);
up = Math.min(n1, up);
if (up <= lowerBound)
return lowerBound;
}
return up;
}
float up2 = up * up;
float lb2 = lowerBound * lowerBound;
for (int i = 0; objects != null && i < objects.size(); i++) {
Vector3f o = (Vector3f) objects.get(i);
float d = o.distanceSquared(x, y, z);
if (d <= lb2)
return lowerBound;
if (d < up2)
up2 = d;
}
return (float) Math.sqrt(up2);
}
}
/**
* Generates Best Candidate samples inside a unit cube.
*
* @author Kai Burjack
*/
public static class Cube {
private int numSamples;
private int numCandidates = 60; // <- use a reasonable default
private long seed;
/**
* Create a new instance of {@link Cube} to configure and generate 'best candidate' sample positions
* on the unit cube with each sample tried <code>numCandidates</code> number of times, and call the
* given <code>callback</code> for each sample generate.
*/
public Cube() {}
/**
* Set the seed to initialize the pseudo-random number generator with.
*
* @param seed
* the seed value
* @return this
*/
public Cube seed(long seed) {
this.seed = seed;
return this;
}
/**
* Set the number of samples to generate.
*
* @param numSamples
* the number of samples
* @return this
*/
public Cube numSamples(int numSamples) {
this.numSamples = numSamples;
return this;
}
/**
* Set the number of candidates to try for each generated sample.
*
* @param numCandidates
* the number of candidates to try
* @return this
*/
public Cube numCandidates(int numCandidates) {
this.numCandidates = numCandidates;
return this;
}
/**
* Generate 'best candidate' sample positions and store the coordinates of all generated samples into the given <code>xyzs</code> float array.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param xyzs
* will hold the x, y and z coordinates of all samples in the order <code>XYZXYZXYZ...</code>.
* This array must have a length of at least <code>numSamples</code>
* @return this
*/
public Cube generate(final float[] xyzs) {
final IntHolder i = new IntHolder();
return generate(new Callback3d() {
public void onNewSample(float x, float y, float z) {
xyzs[3 * i.value + 0] = x;
xyzs[3 * i.value + 1] = y;
xyzs[3 * i.value + 2] = z;
i.value++;
}
});
}
/**
* Generate 'best candidate' sample positions and store the coordinates of all generated samples into the given <code>xyzs</code> FloatBuffer.
* <p>
* The samples will be written starting at the current position of the FloatBuffer. The position of the FloatBuffer will not be modified.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param xyzs
* will hold the x, y and z coordinates of all samples in the order <code>XYZXYZXYZ...</code>.
* This FloatBuffer must have at least <code>numSamples</code> remaining elements.
* The position of the buffer will not be modified by this method
* @return this
*/
public Cube generate(final FloatBuffer xyzs) {
final IntHolder i = new IntHolder();
final int pos = xyzs.position();
return generate(new Callback3d() {
public void onNewSample(float x, float y, float z) {
xyzs.put(pos + 3 * i.value + 0, x);
xyzs.put(pos + 3 * i.value + 1, y);
xyzs.put(pos + 3 * i.value + 2, z);
i.value++;
}
});
}
/**
* Generate 'best candidate' sample positions and call the given <code>callback</code> for each generated sample.
* <p>
* <em>This method performs heap allocations, so should be used sparingly.</em>
*
* @param callback
* will be called with the coordinates of each generated sample position
* @return this
*/
public Cube generate(Callback3d callback) {
Octree octree = new Octree(-1, -1, -1, 2);
Random rnd = new Random(seed);
for (int i = 0; i < numSamples; i++) {
float bestX = 0, bestY = 0, bestZ = 0, bestDist = 0.0f;
for (int c = 0; c < numCandidates; c++) {
float x = rnd.nextFloat() * 2.0f - 1.0f;
float y = rnd.nextFloat() * 2.0f - 1.0f;
float z = rnd.nextFloat() * 2.0f - 1.0f;
float minDist = octree.nearest(x, y, z, bestDist, Float.POSITIVE_INFINITY);
if (minDist > bestDist) {
bestDist = minDist;
bestX = x;
bestY = y;
bestZ = z;
}
}
callback.onNewSample(bestX, bestY, bestZ);
octree.insert(new Vector3f(bestX, bestY, bestZ));
}
return this;
}
}
}