The short answer would be that I’m creating a mesh along a bezier curve. I’m using this method in a couple of areas such as movement boundaries, attack boundaries, terrain boundaries and a slightly different mesh for the move path indicator in which the end does not reconnect with the beginning and the UVs are calculated differently to allow for a scrolling animation.
Creating the mesh is the easy part, creating the spline along which the mesh is created is considerably more complicated for the faction boundaries. The creation of faction boundaries is spawned off in a separate thread to prevent hiccups in the frame rate, after the mesh is created it’s queued up and added to the scene.
Long story short I iterate through a particular faction’s space stations looking for the top most station then look for the top most node(grid space) within that station’s resource collection range and begin looping over the boundary using a list of all nodes owned by this faction. I check to see which space station owns the node I’m currently adding to the boundary and remove it from a list of stations that have not been checked. Once I reach the end I rinse and repeat using the list of stations I was removing from so I don’t leave out any space stations whose boundaries are disconnected from others.
once I have the external boundaries completed and stored in a list of boundaries I iterate through the interior of those boundaries looking for holes and use a similar method to loop over the edges of those holes creating interior boundaries. Once that is done I iterate over those interior boundaries again looking for an occurrence where a node in the inner boundary is connected to another part of the boundary via a single node and separate them into two inner boundaries because it looks weird otherwise data:image/s3,"s3://crabby-images/07666/07666ecd244031cbc79adbe17d0c5f609400c120" alt=":slight_smile: :slight_smile:"
After all of that is done I iterate over all of the boundaries creating lists of control points, splines, based on several factors then pass those splines into a custom mesh class that creates a mesh along the spline.
Here’s the two bezier mesh classes I have right now. They weren’t really developed with versatility in mind so they really only work with a 2D spline that goes along the x/z-axis with normals pointing up along the y-axis.
Cyclical bezier:
public class BezierPathCyclic extends Mesh {
private final Spline spline;
private final float width1;
private final float width2;
public BezierPathCyclic(Spline spline, float width1, float width2) {
super();
this.spline = spline;
this.width1 = width1;
this.width2 = width2;
create();
}
private void create() {
int numSegments = 32;
Vector3f tmp = new Vector3f();
Vector3f tmp2 = new Vector3f();
Node node = new Node("Path Builder");
new Node().attachChild(node);
int numPoints = (spline.getControlPoints().size() + 2) / 3;
int currentControlPoint = 0;
float[] verts = new float[(((numPoints - 1) * numSegments + 1) * 3) * 2];
float[] normals = new float[verts.length];
float[] texCoord = new float[(((numPoints - 1) * numSegments + 1) * 2) * 2];
float[] texCoord2 = new float[texCoord.length];
int index = 0;
int nIndex = 0;
int uvIndex = 0;
int uvIndex2 = 0;
float uvY = 0f;
float uvYSize = 0.04f;
float gameWidth = UI.getGame().getWidth() * CDNode.SIZE;
float gameHeight = UI.getGame().getHeight() * CDNode.SIZE;
for (int i = 0; i < numPoints - 1; i++) {
if (i == 0) {
spline.interpolate(1f / numSegments, 0, tmp);
node.setLocalTranslation(spline.getControlPoints().get(currentControlPoint));
node.lookAt(tmp, Vector3f.UNIT_Y);
node.localToWorld(new Vector3f(width1, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
node.localToWorld(new Vector3f(-width2, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = 1;
texCoord[uvIndex++] = 0;
texCoord2[uvIndex2++] = 1f - (verts[index - 6] / gameWidth);
texCoord2[uvIndex2++] = verts[index - 4] / gameHeight;
texCoord2[uvIndex2++] = 1f - (verts[index - 3] / gameWidth);
texCoord2[uvIndex2++] = verts[index - 1] / gameHeight;
} else {
tmp = node.getLocalTranslation().clone();
node.setLocalTranslation(spline.getControlPoints().get(currentControlPoint));
node.lookAt(tmp, Vector3f.UNIT_Y);
node.localToWorld(new Vector3f(-width1, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
node.localToWorld(new Vector3f(width2, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
uvY += tmp.distance(node.getLocalTranslation()) * uvYSize;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = uvY;
texCoord[uvIndex++] = 1;
texCoord[uvIndex++] = uvY;
texCoord2[uvIndex2++] = 1f - (verts[index - 6] / gameWidth);
texCoord2[uvIndex2++] = verts[index - 4] / gameHeight;
texCoord2[uvIndex2++] = 1f - (verts[index - 3] / gameWidth);
texCoord2[uvIndex2++] = verts[index - 1] / gameHeight;
}
for (int s = 1; s < numSegments; s++) {
tmp = node.getLocalTranslation().clone();
spline.interpolate((float)s / numSegments, currentControlPoint, tmp2);
node.setLocalTranslation(tmp2);
node.lookAt(tmp, Vector3f.UNIT_Y);
node.localToWorld(new Vector3f(-width1, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
node.localToWorld(new Vector3f(width2, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
uvY += tmp.distance(node.getLocalTranslation()) * uvYSize;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = uvY;
texCoord[uvIndex++] = 1;
texCoord[uvIndex++] = uvY;
texCoord2[uvIndex2++] = 1f - (verts[index - 6] / gameWidth);
texCoord2[uvIndex2++] = verts[index - 4] / gameHeight;
texCoord2[uvIndex2++] = 1f - (verts[index - 3] / gameWidth);
texCoord2[uvIndex2++] = verts[index - 1] / gameHeight;
}
currentControlPoint += 3;
}
tmp = node.getLocalTranslation().clone();
node.setLocalTranslation(spline.getControlPoints().get(spline.getControlPoints().size() - 1));
node.lookAt(tmp, Vector3f.UNIT_Y);
verts[index++] = verts[0];
verts[index++] = 0f;
verts[index++] = verts[2];
verts[index++] = verts[3];
verts[index++] = 0f;
verts[index++] = verts[5];
node.removeFromParent();
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
uvY += tmp.distance(node.getLocalTranslation()) * uvYSize;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = uvY;
texCoord[uvIndex++] = 1;
texCoord[uvIndex++] = uvY;
texCoord2[uvIndex2++] = 1f - (verts[index - 6] / gameWidth);
texCoord2[uvIndex2++] = verts[index - 4] / gameHeight;
texCoord2[uvIndex2++] = 1f - (verts[index - 3] / gameWidth);
texCoord2[uvIndex2++] = verts[index - 1] / gameHeight;
short[] indices = new short[((numPoints - 1) * numSegments) * 6];
for (int ind = 0; ind < (numPoints - 1) * numSegments; ind++) {
int indInd = ind * 6;
int indV = ind * 2;
indices[indInd++] = (short)indV;
indices[indInd++] = (short)(indV + 1);
indices[indInd++] = (short)(indV + 2);
indices[indInd++] = (short)(indV + 1);
indices[indInd++] = (short)(indV + 3);
indices[indInd] = (short)(indV + 2);
}
setBuffer(VertexBuffer.Type.Position, 3, verts);
setBuffer(VertexBuffer.Type.Normal, 3, normals);
setBuffer(VertexBuffer.Type.TexCoord, 2, texCoord);
setBuffer(VertexBuffer.Type.TexCoord2, 2, texCoord2);
setBuffer(VertexBuffer.Type.Index, 3, indices);
updateBound();
updateCounts();
}
}
Non cyclical bezier:
public class BezierPath extends Mesh {
private final Spline spline;
public BezierPath(Spline spline) {
super();
this.spline = spline;
create();
}
private void create() {
int numSegments = 64;
Vector3f tmp = new Vector3f();
Vector3f tmp2 = new Vector3f();
Node node = new Node("Path Builder");
new Node().attachChild(node);
int numPoints = (spline.getControlPoints().size() + 2) / 3;
int currentControlPoint = 0;
float[] verts = new float[(((numPoints - 1) * numSegments + 1) * 3) * 2];
float[] normals = new float[verts.length];
float[] texCoord = new float[(((numPoints - 1) * numSegments + 1) * 2) * 2];
float[] texCoord2 = new float[texCoord.length];
int index = 0;
int nIndex = 0;
int uvIndex = 0;
int uvIndex2 = 0;
float uvY = 0f;
float uvYSize = 0.04f;
float width = 4.5f;
for (int i = 0; i < numPoints - 1; i++) {
if (i == 0) {
spline.interpolate(1f / numSegments, 0, tmp);
node.setLocalTranslation(spline.getControlPoints().get(currentControlPoint));
node.lookAt(tmp, Vector3f.UNIT_Y);
node.localToWorld(new Vector3f(width * 0.5f, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
node.localToWorld(new Vector3f(-width * 0.5f, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = 1;
texCoord[uvIndex++] = 0;
texCoord2[uvIndex2++] = 0;
texCoord2[uvIndex2++] = 0;
texCoord2[uvIndex2++] = 1;
texCoord2[uvIndex2++] = 0;
} else {
tmp = node.getLocalTranslation().clone();
node.setLocalTranslation(spline.getControlPoints().get(currentControlPoint));
node.lookAt(tmp, Vector3f.UNIT_Y);
node.localToWorld(new Vector3f(-width, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
node.localToWorld(new Vector3f(width, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
uvY += tmp.distance(node.getLocalTranslation()) * uvYSize;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = uvY;
texCoord[uvIndex++] = 1;
texCoord[uvIndex++] = uvY;
}
for (int s = 1; s < numSegments; s++) {
tmp = node.getLocalTranslation().clone();
spline.interpolate((float)s / numSegments, currentControlPoint, tmp2);
node.setLocalTranslation(tmp2);
node.lookAt(tmp, Vector3f.UNIT_Y);
float wPerc;
if (currentControlPoint == 0) {
wPerc = (((float) s / numSegments) * 0.5f) + 0.5f;
texCoord2[uvIndex2++] = 0;
texCoord2[uvIndex2++] = (float) s / numSegments;
texCoord2[uvIndex2++] = 1;
texCoord2[uvIndex2++] = (float) s / numSegments;
} else if ((currentControlPoint / 3) == numPoints - 2) {
wPerc = (0.5f - (((float) s / numSegments) * 0.5f)) + 0.5f;
texCoord2[uvIndex2++] = 0;
texCoord2[uvIndex2++] = 1f - ((float) s / numSegments);
texCoord2[uvIndex2++] = 1;
texCoord2[uvIndex2++] = 1f - ((float) s / numSegments);
} else {
wPerc = 1f;
texCoord2[uvIndex2++] = 0;
texCoord2[uvIndex2++] = 1;
texCoord2[uvIndex2++] = 1;
texCoord2[uvIndex2++] = 1;
}
node.localToWorld(new Vector3f(-width * wPerc, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
node.localToWorld(new Vector3f(width * wPerc, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
uvY += tmp.distance(node.getLocalTranslation()) * uvYSize;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = uvY;
texCoord[uvIndex++] = 1;
texCoord[uvIndex++] = uvY;
}
currentControlPoint += 3;
}
tmp = node.getLocalTranslation().clone();
node.setLocalTranslation(spline.getControlPoints().get(spline.getControlPoints().size() - 1));
node.lookAt(tmp, Vector3f.UNIT_Y);
node.localToWorld(new Vector3f(-width * 0.5f, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
node.localToWorld(new Vector3f(width * 0.5f, 0f, 0f), tmp2);
verts[index++] = tmp2.x;
verts[index++] = 0f;
verts[index++] = tmp2.z;
node.removeFromParent();
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
normals[nIndex++] = 0;
normals[nIndex++] = 1;
normals[nIndex++] = 0;
uvY += tmp.distance(node.getLocalTranslation()) * uvYSize;
texCoord[uvIndex++] = 0;
texCoord[uvIndex++] = uvY;
texCoord[uvIndex++] = 1;
texCoord[uvIndex++] = uvY;
texCoord2[uvIndex2++] = 0;
texCoord2[uvIndex2++] = 0;
texCoord2[uvIndex2++] = 1;
texCoord2[uvIndex2++] = 0;
short[] indices = new short[((numPoints - 1) * numSegments) * 6];
for (int ind = 0; ind < (numPoints - 1) * numSegments; ind++) {
int indInd = ind * 6;
int indV = ind * 2;
indices[indInd++] = (short)indV;
indices[indInd++] = (short)(indV + 1);
indices[indInd++] = (short)(indV + 2);
indices[indInd++] = (short)(indV + 1);
indices[indInd++] = (short)(indV + 3);
indices[indInd] = (short)(indV + 2);
}
setBuffer(VertexBuffer.Type.Position, 3, verts);
setBuffer(VertexBuffer.Type.Normal, 3, normals);
setBuffer(VertexBuffer.Type.TexCoord, 2, texCoord);
setBuffer(VertexBuffer.Type.TexCoord2, 2, texCoord2);
setBuffer(VertexBuffer.Type.Index, 3, indices);
updateBound();
updateCounts();
}
}
P.S. The second set of UV coordinates on the cyclical bezier are used for an alpha mask using world coordinates in the shader. The alpha mask is the fog of war texture which is updated only on frames in which the state of the fog of war is changed.
When visibility of a particular area is changed a viewport is enabled, takes a single frame snapshot of a black and white scene rendered to an off-screen buffer which is then blurred over a few passes after which the viewport is disabled. The resolution of the texture depends on the size of the level, but it’s not very big. Blurring it adds a nice fade effect around the edges, but also prevents the texture from looking blocky when scaled up.