mirror of
https://github.com/jhasse/poly2tri.git
synced 2024-12-31 23:23:30 +01:00
fixed triangle location bug
This commit is contained in:
parent
6517956535
commit
324fb8207e
@ -124,12 +124,12 @@ class CDT(val points: List[Point], val segments: List[Segment], iTriangle: Trian
|
|||||||
|
|
||||||
// Implement sweep-line
|
// Implement sweep-line
|
||||||
private def sweep {
|
private def sweep {
|
||||||
for(i <- 1 until 10 /*points.size*/) {
|
for(i <- 1 until 11 /*points.size*/) {
|
||||||
val point = points(i)
|
val point = points(i)
|
||||||
// Process Point event
|
// Process Point event
|
||||||
val triangle = pointEvent(point)
|
val triangle = pointEvent(point)
|
||||||
// Process edge events
|
// Process edge events
|
||||||
point.edges.foreach(e => edgeEvent(e, triangle))
|
point.edges.foreach(e => if(!triangle.contains(e)) edgeEvent(e, triangle))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,31 +137,60 @@ class CDT(val points: List[Point], val segments: List[Segment], iTriangle: Trian
|
|||||||
private def pointEvent(point: Point): Triangle = {
|
private def pointEvent(point: Point): Triangle = {
|
||||||
|
|
||||||
val node = aFront.locate(point)
|
val node = aFront.locate(point)
|
||||||
|
var newNode: Node = null
|
||||||
|
|
||||||
// Neightbor points (ccw & cw) and triangle(i)
|
if(point.x == node.point.x) {
|
||||||
val cwPoint = node.next.point
|
|
||||||
val ccwPoint = node.point
|
|
||||||
val nTri = node.triangle
|
|
||||||
|
|
||||||
val pts = Array(point, ccwPoint, cwPoint)
|
// Projected point coincides with existing point; create two triangles
|
||||||
val neighbors = Array(nTri, null, null)
|
val rPts = Array(point, node.point, node.next.point)
|
||||||
val triangle = new Triangle(pts, neighbors)
|
val rNeighbors = Array(node.triangle, null, null)
|
||||||
mesh.map += triangle
|
val rTriangle = new Triangle(rPts, rNeighbors)
|
||||||
|
|
||||||
// Check if edges need to be swapped to preserve CDT
|
val lPts = Array(node.prev.point, node.point, point)
|
||||||
// TODO: Make sure AFront pointers are updated correctly
|
val lNeighbors = Array(rTriangle, null, node.prev.triangle)
|
||||||
val oPoint = nTri oppositePoint triangle
|
val lTriangle = new Triangle(lPts, lNeighbors)
|
||||||
if(illegal(ccwPoint, oPoint, cwPoint, point)) {
|
|
||||||
legalization(triangle, nTri, oPoint)
|
|
||||||
}
|
|
||||||
|
|
||||||
nTri.updateNeighbors(ccwPoint, cwPoint, triangle)
|
rTriangle.neighbors(2) = lTriangle
|
||||||
|
mesh.map += lTriangle
|
||||||
|
mesh.map += rTriangle
|
||||||
|
|
||||||
|
// Legalize new triangles
|
||||||
|
legalization(rTriangle, rTriangle.neighbors(0))
|
||||||
|
legalization(lTriangle, lTriangle.neighbors(2))
|
||||||
|
|
||||||
|
// Update neighbor pointers
|
||||||
|
node.triangle.updateNeighbors(node.point, node.next.point, lTriangle)
|
||||||
|
node.prev.triangle.updateNeighbors(node.point, node.prev.point, rTriangle)
|
||||||
|
|
||||||
|
// Update advancing front
|
||||||
|
newNode = aFront.insert(point, rTriangle, node)
|
||||||
|
aFront -= (node.prev, node, rTriangle)
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
// Projected point hits advancing front; create new triangle
|
||||||
|
val cwPoint = node.next.point
|
||||||
|
val ccwPoint = node.point
|
||||||
|
val nTri = node.triangle
|
||||||
|
|
||||||
|
val pts = Array(point, ccwPoint, cwPoint)
|
||||||
|
val neighbors = Array(nTri, null, null)
|
||||||
|
val triangle = new Triangle(pts, neighbors)
|
||||||
|
mesh.map += triangle
|
||||||
|
|
||||||
|
// Legalize
|
||||||
|
legalization(triangle, nTri)
|
||||||
|
// Update neighbor pointers
|
||||||
|
nTri.updateNeighbors(ccwPoint, cwPoint, triangle)
|
||||||
|
// Update advancing front
|
||||||
|
newNode = aFront.insert(point, triangle, node)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
// Update advancing front
|
|
||||||
val newNode = aFront.insert(point, triangle, node)
|
|
||||||
// Fill in adjacent triangles if required
|
// Fill in adjacent triangles if required
|
||||||
scan(newNode)
|
scan(newNode)
|
||||||
triangle
|
newNode.triangle
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// EdgeEvent
|
// EdgeEvent
|
||||||
@ -169,7 +198,6 @@ class CDT(val points: List[Point], val segments: List[Segment], iTriangle: Trian
|
|||||||
|
|
||||||
// STEP 1: Locate the first intersected triangle
|
// STEP 1: Locate the first intersected triangle
|
||||||
val firstTriangle = triangle.locateFirst(edge)
|
val firstTriangle = triangle.locateFirst(edge)
|
||||||
|
|
||||||
// STEP 2: Remove intersected triangles
|
// STEP 2: Remove intersected triangles
|
||||||
if(firstTriangle != null && !firstTriangle.contains(edge)) {
|
if(firstTriangle != null && !firstTriangle.contains(edge)) {
|
||||||
// Collect intersected triangles
|
// Collect intersected triangles
|
||||||
@ -182,21 +210,21 @@ class CDT(val points: List[Point], val segments: List[Segment], iTriangle: Trian
|
|||||||
//triangles.foreach(t => mesh.map -= t)
|
//triangles.foreach(t => mesh.map -= t)
|
||||||
} else if(firstTriangle == null) {
|
} else if(firstTriangle == null) {
|
||||||
|
|
||||||
// Traverse the AFront, and build triangle list
|
// Traverse the AFront, and build triangles
|
||||||
var node = aFront.locate(edge.q)
|
var node = aFront.locate(edge.q)
|
||||||
node = node.next
|
node = node.next
|
||||||
val points = new ArrayBuffer[Point]
|
val points = new ArrayBuffer[Point]
|
||||||
|
|
||||||
if(edge.p.x > edge.q.x) {
|
if(edge.p.x > edge.q.x) {
|
||||||
// Search right
|
// Search right
|
||||||
while(node.point != edge.p) {
|
while(node != null && node.point != edge.p) {
|
||||||
// Collect points
|
// Collect points
|
||||||
points += node.point
|
points += node.point
|
||||||
node = node.next
|
node = node.next
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Search left
|
// Search left
|
||||||
while(node.point != edge.p) {
|
while(node != null && node.point != edge.p) {
|
||||||
// Collect points
|
// Collect points
|
||||||
points += node.point
|
points += node.point
|
||||||
node = node.prev
|
node = node.prev
|
||||||
@ -209,6 +237,7 @@ class CDT(val points: List[Point], val segments: List[Segment], iTriangle: Trian
|
|||||||
}
|
}
|
||||||
|
|
||||||
// STEP 3: Triangulate empty areas.
|
// STEP 3: Triangulate empty areas.
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Marc Vigo Anglada's triangulatePseudopolygonDelaunay algo
|
// Marc Vigo Anglada's triangulatePseudopolygonDelaunay algo
|
||||||
@ -293,22 +322,25 @@ class CDT(val points: List[Point], val segments: List[Segment], iTriangle: Trian
|
|||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
// Flip edges and rotate everything clockwise
|
// Ensure new adjacent triangles are legal
|
||||||
private def legalization(t1: Triangle, t2: Triangle, oPoint: Point) {
|
private def legalization(t1: Triangle, t2: Triangle) {
|
||||||
// Rotate points
|
val oPoint = t2 oppositePoint t1
|
||||||
val point = t1.points(0)
|
if(illegal(t1.points(1), oPoint, t1.points(2), t1.points(0))) {
|
||||||
t1.points(1) = t1.points(0)
|
// Flip edges and rotate everything clockwise
|
||||||
t1.points(0) = t1.points(2)
|
val point = t1.points(0)
|
||||||
t1.points(2) = oPoint
|
t1.points(1) = t1.points(0)
|
||||||
val tmp = t2.points(1)
|
t1.points(0) = t1.points(2)
|
||||||
t2.points(1) = point
|
t1.points(2) = oPoint
|
||||||
t2.points(0) = t2.points(2)
|
val tmp = t2.points(1)
|
||||||
t2.points(2) = tmp
|
t2.points(1) = point
|
||||||
|
t2.points(0) = t2.points(2)
|
||||||
|
t2.points(2) = tmp
|
||||||
|
|
||||||
t1.updatePoints
|
t1.updatePoints
|
||||||
t2.updatePoints
|
t2.updatePoints
|
||||||
// TODO: Rotate neighbors
|
// TODO: Rotate neighbors
|
||||||
println("legalize")
|
println("legalize")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private def finalization {
|
private def finalization {
|
||||||
|
@ -54,9 +54,9 @@ class Triangle(val points: Array[Point], val neighbors: Array[Triangle]) {
|
|||||||
|
|
||||||
// Update neighbor pointers
|
// Update neighbor pointers
|
||||||
def updateNeighbors(ccwPoint: Point, cwPoint: Point, triangle: Triangle) {
|
def updateNeighbors(ccwPoint: Point, cwPoint: Point, triangle: Triangle) {
|
||||||
if(ccwPoint == points(2) && cwPoint == points(1))
|
if((ccwPoint == points(2) && cwPoint == points(1)) || (ccwPoint == points(1) && cwPoint == points(2)))
|
||||||
neighbors(0) = triangle
|
neighbors(0) = triangle
|
||||||
else if(ccwPoint == points(0) && cwPoint == points(2))
|
else if((ccwPoint == points(0) && cwPoint == points(2)) || (ccwPoint == points(0) && cwPoint == points(2)))
|
||||||
neighbors(1) = triangle
|
neighbors(1) = triangle
|
||||||
else
|
else
|
||||||
neighbors(2) = triangle
|
neighbors(2) = triangle
|
||||||
@ -91,7 +91,7 @@ class Triangle(val points: Array[Point], val neighbors: Array[Triangle]) {
|
|||||||
|
|
||||||
def locateFirst(edge: Segment): Triangle = {
|
def locateFirst(edge: Segment): Triangle = {
|
||||||
val p = edge.p
|
val p = edge.p
|
||||||
if(contains(p)) return this
|
if(contains(p) || contains(edge)) return this
|
||||||
val q = edge.q
|
val q = edge.q
|
||||||
val e = p - q
|
val e = p - q
|
||||||
if(q == points(0)) {
|
if(q == points(0)) {
|
||||||
|
Loading…
Reference in New Issue
Block a user