/**
* Copyright 2008 - 2015 The Loon Game Engine Authors
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*
* @project loon
* @author cping
* @email:javachenpeng@yahoo.com
* @version 0.5
*/
package loon.geom;
public class TriangleNeat implements Triangle {
/**
*
*/
private static final long serialVersionUID = 1L;
private static final float EPSILON = 1E-006F;
private float[] pointsX;
private float[] pointsY;
private int numPoints;
private Edge[] edges;
private int[] sV;
private int numEdges;
private Triangle[] triangles;
private int numTriangles;
private float offset = EPSILON;
public TriangleNeat() {
pointsX = new float[100];
pointsY = new float[100];
numPoints = 0;
edges = new Edge[100];
numEdges = 0;
triangles = new Triangle[100];
numTriangles = 0;
}
public void clear() {
numPoints = 0;
numEdges = 0;
numTriangles = 0;
}
private int findEdge(int i, int j) {
int k;
int l;
if (i < j) {
k = i;
l = j;
} else {
k = j;
l = i;
}
for (int i1 = 0; i1 < numEdges; i1++) {
if (edges[i1].v0 == k && edges[i1].v1 == l) {
return i1;
}
}
return -1;
}
private void addEdge(int i, int j, int k) {
int l1 = findEdge(i, j);
int j1;
int k1;
Edge edge;
if (l1 < 0) {
if (numEdges == edges.length) {
Edge aedge[] = new Edge[edges.length * 2];
System.arraycopy(edges, 0, aedge, 0, numEdges);
edges = aedge;
}
j1 = -1;
k1 = -1;
l1 = numEdges++;
edge = edges[l1] = new Edge();
} else {
edge = edges[l1];
j1 = edge.t0;
k1 = edge.t1;
}
int l;
int i1;
if (i < j) {
l = i;
i1 = j;
j1 = k;
} else {
l = j;
i1 = i;
k1 = k;
}
edge.v0 = l;
edge.v1 = i1;
edge.t0 = j1;
edge.t1 = k1;
edge.suspect = true;
}
void markSuspect(int i, int j, boolean flag) throws Exception {
int k;
if (0 > (k = findEdge(i, j))) {
throw new Exception("Attempt to mark unknown edge");
} else {
edges[k].suspect = flag;
return;
}
}
private static boolean insideTriangle(float f, float f1, float f2,
float f3, float f4, float f5, float f6, float f7) {
float f8 = f4 - f2;
float f9 = f5 - f3;
float f10 = f - f4;
float f11 = f1 - f5;
float f12 = f2 - f;
float f13 = f3 - f1;
float f14 = f6 - f;
float f15 = f7 - f1;
float f16 = f6 - f2;
float f17 = f7 - f3;
float f18 = f6 - f4;
float f19 = f7 - f5;
float f22 = f8 * f17 - f9 * f16;
float f20 = f12 * f15 - f13 * f14;
float f21 = f10 * f19 - f11 * f18;
return f22 >= 0.0D && f21 >= 0.0D && f20 >= 0.0D;
}
private boolean snip(int i, int j, int k, int l) {
float f = pointsX[sV[i]];
float f1 = pointsY[sV[i]];
float f2 = pointsX[sV[j]];
float f3 = pointsY[sV[j]];
float f4 = pointsX[sV[k]];
float f5 = pointsY[sV[k]];
if (1E-006F > (f2 - f) * (f5 - f1) - (f3 - f1) * (f4 - f))
return false;
for (int i1 = 0; i1 < l; i1++)
if (i1 != i && i1 != j && i1 != k) {
float f6 = pointsX[sV[i1]];
float f7 = pointsY[sV[i1]];
if (insideTriangle(f, f1, f2, f3, f4, f5, f6, f7))
return false;
}
return true;
}
private float area() {
float f = 0.0F;
int i = numPoints - 1;
for (int j = 0; j < numPoints;) {
f += pointsX[i] * pointsY[j] - pointsY[i] * pointsX[j];
i = j++;
}
return f * 0.5F;
}
public void basicTriangulation() throws Exception {
int i = numPoints;
if (i < 3)
return;
numEdges = 0;
numTriangles = 0;
sV = new int[i];
if (0.0D < area()) {
for (int k = 0; k < i; k++)
sV[k] = k;
} else {
for (int l = 0; l < i; l++)
sV[l] = numPoints - 1 - l;
}
int k1 = 2 * i;
int i1 = i - 1;
while (i > 2) {
if (0 >= k1--) {
throw new Exception("Bad polygon");
}
int j = i1;
if (i <= j)
j = 0;
i1 = j + 1;
if (i <= i1)
i1 = 0;
int j1 = i1 + 1;
if (i <= j1)
j1 = 0;
if (snip(j, i1, j1, i)) {
int l1 = sV[j];
int i2 = sV[i1];
int j2 = sV[j1];
if (numTriangles == triangles.length) {
Triangle atriangle[] = new Triangle[triangles.length * 2];
System.arraycopy(triangles, 0, atriangle, 0, numTriangles);
triangles = atriangle;
}
triangles[numTriangles] = new Triangle(l1, i2, j2);
addEdge(l1, i2, numTriangles);
addEdge(i2, j2, numTriangles);
addEdge(j2, l1, numTriangles);
numTriangles++;
int k2 = i1;
for (int l2 = i1 + 1; l2 < i; l2++) {
sV[k2] = sV[l2];
k2++;
}
i--;
k1 = 2 * i;
}
}
sV = null;
}
public boolean triangulate() {
try {
basicTriangulation();
return true;
} catch (Exception e) {
numEdges = 0;
}
return false;
}
public void addPolyPoint(float x, float y) {
for (int i = 0; i < numPoints; i++) {
if ((pointsX[i] == x) && (pointsY[i] == y)) {
y += offset;
offset += EPSILON;
}
}
if (numPoints == pointsX.length) {
float af[] = new float[numPoints * 2];
System.arraycopy(pointsX, 0, af, 0, numPoints);
pointsX = af;
af = new float[numPoints * 2];
System.arraycopy(pointsY, 0, af, 0, numPoints);
pointsY = af;
}
pointsX[numPoints] = x;
pointsY[numPoints] = y;
numPoints++;
}
class Triangle {
int[] v;
Triangle(int i, int j, int k) {
v = new int[3];
v[0] = i;
v[1] = j;
v[2] = k;
}
}
class Edge {
int v0;
int v1;
int t0;
int t1;
boolean suspect;
Edge() {
v0 = -1;
v1 = -1;
t0 = -1;
t1 = -1;
}
}
public int getTriangleCount() {
return numTriangles;
}
public float[] getTrianglePoint(int tri, int i) {
float xp = pointsX[triangles[tri].v[i]];
float yp = pointsY[triangles[tri].v[i]];
return new float[] { xp, yp };
}
public void startHole() {
}
}