Skip to content

Commit

Permalink
#7 Code refactoring and cleanup
Browse files Browse the repository at this point in the history
  • Loading branch information
artem-ogre committed Oct 6, 2023
1 parent 1167f7b commit 8687b06
Show file tree
Hide file tree
Showing 4 changed files with 84 additions and 94 deletions.
2 changes: 2 additions & 0 deletions CDT/include/CDTUtils.h
Original file line number Diff line number Diff line change
Expand Up @@ -466,6 +466,8 @@ sineOfSmallestAngle(const V2d<T>& a, const V2d<T>& b, const V2d<T>& c);
template <typename T>
CDT_EXPORT T smallestAngle(const V2d<T>& a, const V2d<T>& b, const V2d<T>& c);

CDT_EXPORT bool touchesSuperTriangle(const Triangle& t);

} // namespace CDT

#ifndef CDT_USE_AS_COMPILED_LIBRARY
Expand Down
5 changes: 5 additions & 0 deletions CDT/include/CDTUtils.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -368,4 +368,9 @@ T smallestAngle(const V2d<T>& a, const V2d<T>& b, const V2d<T>& c)
return std::asin(angleSine);
}

bool touchesSuperTriangle(const Triangle& t)
{
return t.vertices[0] < 3 || t.vertices[1] < 3 || t.vertices[2] < 3;
}

} // namespace CDT
22 changes: 8 additions & 14 deletions CDT/include/Triangulation.h
Original file line number Diff line number Diff line change
Expand Up @@ -512,16 +512,14 @@ class CDT_EXPORT Triangulation
const Triangle& tri,
RefinementCriterion::Enum refinementCriterion,
T refinementThreshold) const;
/// Search in all fixed edges to find encroached edges, each fixed edge is
/// checked against its opposite vertices
/// Returns queue of encroached edges
EdgeQueue detectEncroachedEdges();
/// Search in all fixed edges to find encroached edges, each fixed edge is
/// checked against its opposite vertices and vertex v
/// Returns queue of encroached edges
EdgeQueue detectEncroachedEdges(const V2d<T>& v);
/// Check if edge is encroached by its opposed vertices
bool isEdgeEncroached(const Edge& edge) const;
bool isEdgeEncroachedBy(const Edge& edge, const V2d<T>& v) const;
/// Find all fixed edges encroached by its opposed vertices
EdgeQueue allEncroachedEdges() const;
/// Find all fixed edges encroached by a given vertex
EdgeQueue edgesEncroachedBy(const V2d<T>& v) const;
/// Recursively split encroached edges
/// @return vector of badly shaped triangles
TriIndVec resolveEncroachedEdges(
EdgeQueue encroachedEdges,
VertInd& newVertBudget,
Expand All @@ -530,11 +528,7 @@ class CDT_EXPORT Triangulation
RefinementCriterion::Enum refinementCriterion =
RefinementCriterion::SmallestAngle,
T badTriangleThreshold = T(0));
VertInd splitEncroachedEdge(
Edge e,
TriInd iT,
TriInd iTopo,
VertInd steinerVerticesOffset);
VertInd splitEncroachedEdge(Edge edge, VertInd steinerVerticesOffset);
void changeNeighbor(TriInd iT, TriInd oldNeighbor, TriInd newNeighbor);
void changeNeighbor(
TriInd iT,
Expand Down
149 changes: 69 additions & 80 deletions CDT/include/Triangulation.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -1250,25 +1250,39 @@ bool Triangulation<T, TNearPointLocator>::isRefinementNeeded(
}

template <typename T, typename TNearPointLocator>
EdgeQueue Triangulation<T, TNearPointLocator>::detectEncroachedEdges()
bool Triangulation<T, TNearPointLocator>::isEdgeEncroached(
const Edge& edge) const
{
// Search in all fixed edges to find encroached edges, each fixed edge is
// checked against its opposite vertices
TriInd iT, iTopo;
std::tie(iT, iTopo) = edgeTriangles(edge.v1(), edge.v2());
assert(iT != invalidIndex && iTopo != invalidIndex);
const VertInd v1 = opposedVertex(triangles[iT], iTopo);
const VertInd v2 = opposedVertex(triangles[iTopo], iT);
const V2d<T>& edgeStart = vertices[edge.v1()];
const V2d<T>& edgeEnd = vertices[edge.v2()];
return isEncroachingOnEdge(vertices[v1], edgeStart, edgeEnd) ||
isEncroachingOnEdge(vertices[v2], edgeStart, edgeEnd);
}

template <typename T, typename TNearPointLocator>
bool Triangulation<T, TNearPointLocator>::isEdgeEncroachedBy(
const Edge& edge,
const V2d<T>& v) const
{
return isEncroachingOnEdge(v, vertices[edge.v1()], vertices[edge.v2()]);
}

template <typename T, typename TNearPointLocator>
EdgeQueue Triangulation<T, TNearPointLocator>::allEncroachedEdges() const
{
// Search in all fixed edges to find encroached edges
// Returns queue of encroached edges
EdgeQueue encroachedEdges;
typedef EdgeUSet::const_iterator Iter;
for(Iter it = fixedEdges.begin(); it != fixedEdges.end(); ++it)
{
const Edge edge = *it;
TriInd iT, iTopo;
std::tie(iT, iTopo) = edgeTriangles(edge.v1(), edge.v2());
assert(iT != invalidIndex && iTopo != invalidIndex);
const VertInd v1 = opposedVertex(triangles[iT], iTopo);
const VertInd v2 = opposedVertex(triangles[iTopo], iT);
const V2d<T>& edgeStart = vertices[edge.v1()];
const V2d<T>& edgeEnd = vertices[edge.v2()];
if(isEncroachingOnEdge(vertices[v1], edgeStart, edgeEnd) ||
isEncroachingOnEdge(vertices[v2], edgeStart, edgeEnd))
if(isEdgeEncroached(edge))
{
encroachedEdges.push(edge);
}
Expand All @@ -1278,19 +1292,17 @@ EdgeQueue Triangulation<T, TNearPointLocator>::detectEncroachedEdges()

template <typename T, typename TNearPointLocator>
EdgeQueue
Triangulation<T, TNearPointLocator>::detectEncroachedEdges(const V2d<T>& v)
Triangulation<T, TNearPointLocator>::edgesEncroachedBy(const V2d<T>& v) const
{
// Search in all fixed edges to find encroached edges, each fixed edge is
// checked against its opposite vertices and vertex v
// Returns queue of encroached edges
// Search in all fixed edges to find edges encroached by v, each fixed edge
// is checked vertex v Returns queue of encroached edges
EdgeQueue encroachedEdges;
typedef EdgeUSet::const_iterator Iter;
for(Iter it = fixedEdges.begin(); it != fixedEdges.end(); ++it)
{
const Edge edge = *it;
if(isEncroachingOnEdge(v, vertices[edge.v1()], vertices[edge.v2()]))
if(isEdgeEncroachedBy(*it, v))
{
encroachedEdges.push(edge);
encroachedEdges.push(*it);
}
}
return encroachedEdges;
Expand All @@ -1312,74 +1324,62 @@ TriIndVec Triangulation<T, TNearPointLocator>::resolveEncroachedEdges(
{
const Edge edge = encroachedEdges.front();
encroachedEdges.pop();
if(!hasEdge(edge.v1(), edge.v2()))
if(fixedEdges.find(edge) == fixedEdges.end())
{
continue;
}
TriInd iT, iTopo;
std::tie(iT, iTopo) = edgeTriangles(edge.v1(), edge.v2());
assert(iT != invalidIndex && iTopo != invalidIndex);
const VertInd i =
splitEncroachedEdge(edge, iT, iTopo, steinerVerticesOffset);
// split encroached edge
const VertInd iSplitVert =
splitEncroachedEdge(edge, steinerVerticesOffset);
--remainingVertexBudget;

const TriInd start = m_vertTris[i];
TriInd currTri = start;
const TriInd start = m_vertTris[iSplitVert];
TriInd iT = start;
do
{
const Triangle& t = triangles[currTri];
const Triangle& t = triangles[iT];
if(circumcenterOrNull &&
isRefinementNeeded(t, refinementCriterion, badTriangleThreshold))
{
badTriangles.push_back(currTri);
badTriangles.push_back(iT);
}
for(int i = 0; i < 3; ++i)
for(Index i(0); i < Index(3); ++i)
{
const Edge edge(t.vertices[i], t.vertices[cw(i)]);
if(fixedEdges.find(edge) == fixedEdges.end())
const Edge triEdge(t.vertices[i], t.vertices[cw(i)]);
if(fixedEdges.find(triEdge) == fixedEdges.end())
continue;
const TriInd iT = currTri;
const TriInd iTopo =
edgeNeighbor(triangles[iT], edge.v1(), edge.v2());
const Triangle& tOpo = triangles[iTopo];
VertInd v1 = opposedVertex(t, iTopo);
VertInd v2 = opposedVertex(tOpo, iT);
const V2d<T>& edgeStart = vertices[edge.v1()];
const V2d<T>& edgeEnd = vertices[edge.v2()];
if(isEncroachingOnEdge(vertices[v1], edgeStart, edgeEnd) ||
isEncroachingOnEdge(vertices[v2], edgeStart, edgeEnd) ||
if(isEdgeEncroached(triEdge) ||
(circumcenterOrNull &&
isEncroachingOnEdge(
*circumcenterOrNull, edgeStart, edgeEnd)))
isEdgeEncroachedBy(triEdge, *circumcenterOrNull)))
{
encroachedEdges.push(edge);
encroachedEdges.push(triEdge);
}
}
currTri = t.next(i).first;
} while(currTri != start);
iT = t.next(iSplitVert).first;
} while(iT != start);
}
return badTriangles;
}

template <typename T, typename TNearPointLocator>
VertInd Triangulation<T, TNearPointLocator>::splitEncroachedEdge(
const Edge splitEdge,
const TriInd iT,
const TriInd iTopo,
const Edge edge,
const VertInd steinerVerticesOffset)
{
TriInd iT, iTopo;
std::tie(iT, iTopo) = edgeTriangles(edge.v1(), edge.v2());
assert(iT != invalidIndex && iTopo != invalidIndex);
const VertInd iMid = static_cast<VertInd>(vertices.size());
const V2d<T>& start = vertices[splitEdge.v1()];
const V2d<T>& end = vertices[splitEdge.v2()];
const V2d<T>& start = vertices[edge.v1()];
const V2d<T>& end = vertices[edge.v2()];
T split = T(0.5);
// check if any of the split edge vertices are Steiner vertices
if(splitEdge.v1() >= steinerVerticesOffset ||
splitEdge.v2() >= steinerVerticesOffset)
if(edge.v1() >= steinerVerticesOffset || edge.v2() >= steinerVerticesOffset)
{
// In Ruppert's paper, he used D(0.01) factor to divide edge length, but
// that introduces FP rounding erros, so it's avoided.
const T len = distance(start, end);
const T d = T(0.5) * len;
const T d = len / T(2);
// Find the splitting distance
T nearestPowerOfTwo = T(1);
while(d > nearestPowerOfTwo)
Expand All @@ -1392,29 +1392,29 @@ VertInd Triangulation<T, TNearPointLocator>::splitEncroachedEdge(
}
assert(abs(nearestPowerOfTwo - pow(2, round(log(d) / log(2.0)))) < 1e6);
split = nearestPowerOfTwo / len;
if(splitEdge.v1() >= steinerVerticesOffset)
if(edge.v1() >= steinerVerticesOffset)
split = T(1) - split;
}
const V2d<T> mid = V2d<T>::make(
detail::lerp(start.x, end.x, split),
detail::lerp(start.y, end.y, split));

// split constraint edge that already exists in triangulation
if(fixedEdges.find(splitEdge) != fixedEdges.end())
if(fixedEdges.find(edge) != fixedEdges.end())
{
const Edge half1(splitEdge.v1(), iMid);
const Edge half2(iMid, splitEdge.v2());
const BoundaryOverlapCount overlaps = overlapCount[splitEdge];
const Edge half1(edge.v1(), iMid);
const Edge half2(iMid, edge.v2());
const BoundaryOverlapCount overlaps = overlapCount[edge];
// remove the edge that will be split
fixedEdges.erase(splitEdge);
overlapCount.erase(splitEdge);
fixedEdges.erase(edge);
overlapCount.erase(edge);
// add split edge's halves
fixEdge(half1, overlaps);
fixEdge(half2, overlaps);
// maintain piece-to-original mapping
EdgeVec newOriginals(1, splitEdge);
EdgeVec newOriginals(1, edge);
const unordered_map<Edge, EdgeVec>::const_iterator originalsIt =
pieceToOriginals.find(splitEdge);
pieceToOriginals.find(edge);
if(originalsIt != pieceToOriginals.end())
{ // edge being split was split before: pass-through originals
newOriginals = originalsIt->second;
Expand Down Expand Up @@ -2245,27 +2245,16 @@ void Triangulation<T, TNearPointLocator>::refineTriangles(
VertInd remainingVertexBudget = maxVerticesToInsert;
const VertInd steinerVerticesOffset = vertices.size();
resolveEncroachedEdges(
detectEncroachedEdges(), remainingVertexBudget, steinerVerticesOffset);
allEncroachedEdges(), remainingVertexBudget, steinerVerticesOffset);

std::queue<TriInd> badTriangles;
for(TriInd iT(0), n = triangles.size(); iT < n; ++iT)
{
const Triangle& t = triangles[iT];
if(t.vertices[0] < 3 || t.vertices[1] < 3 || t.vertices[2] < 3)
continue;

if(isRefinementNeeded(t, refinementCriterion, refinementThreshold))
if(!touchesSuperTriangle(t) &&
isRefinementNeeded(t, refinementCriterion, refinementThreshold))
{
const V2d<T> vert = circumcenter(
vertices[t.vertices[0]],
vertices[t.vertices[1]],
vertices[t.vertices[2]]);
if(locatePointTriangle(
vert, vertices[0], vertices[1], vertices[2]) !=
PtTriLocation::Outside)
{
badTriangles.push(iT);
}
badTriangles.push(iT);
}
}

Expand All @@ -2290,7 +2279,7 @@ void Triangulation<T, TNearPointLocator>::refineTriangles(
}

const TriIndVec badTris = resolveEncroachedEdges(
detectEncroachedEdges(vert),
edgesEncroachedBy(vert),
remainingVertexBudget,
steinerVerticesOffset,
&vert,
Expand Down

0 comments on commit 8687b06

Please sign in to comment.