In the first version of my Cross Sections I figured out how to find intersections between a mesh and a plane. After almost five months I finally found a way to sort these intersections so they can be used to create one or more curves.
In the previous version intersections were calculated for each edge of each polygonal face resulting in hundreds of line segments without any connection. This method wasn’t very efficitent. Every non-boundary edge has 2 adjacent faces, which means the program calculated almost twice as many intersections as necessary.
The new algorithm is actually quite simple and has two phases. First intersections are calculated for individual edges of the mesh. Whenever the intersection is succesful the program stores edge ID, intersection point, parameter and whether the intersection lies on a boundary.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 |
std::set<unsigned int> m_intersectingEdges; std::map<unsigned int, bool> m_isEdgeOnBoundary; std::map<unsigned int, double> m_intersectionParameters; std::map<unsigned int, MPoint> m_intersectionPoints; MStatus findIntersections(MObject& mesh, MMatrix& transform){ MStatus status; MItMeshEdge itEdge(mesh, &status); for (itEdge.reset(); !itEdge.isDone(); itEdge.next()) { double parameter; MPoint intersection; MPoint pointA = itEdge.point(0)*transform; MPoint pointB = itEdge.point(1)*transform; MVector direction = pointB - pointA; if (!intersect(pointA, direction, intersection, parameter)) continue; unsigned int index = itEdge.index(); m_intersectingEdges.insert(index); m_isEdgeOnBoundary[index] = itEdge.onBoundary(); m_intersectionParameters[index] = parameter; m_intersectionPoints[index] = intersection; } return MS::kSuccess; } |
Then comes the second phase which sorts the intersections and creates one or more poly-lines. If there’s an intersection on a boundary, the iterator will make it the beginning of the new curve. In that case the result will be an opened section curve. If none of the intersections lies on a boundary, the first point in the intersection list will be used. In that case the points will form a closed curve.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 |
MStatus generatePlaneSections(MObject &mesh, MMatrix &transform, MObjectArray §ionCurves) { MStatus status; // First stage status = findIntersections(mesh, transform); // Second stage MItMeshPolygon itPolygon(mesh); MItMeshEdge itEdge(mesh); MItMeshVertex itVertex(mesh); while (0<m_intersectingEdges.size()) { auto firstEdge = *m_intersectingEdges.begin(); for (auto iter : m_intersectingEdges) if (m_isEdgeOnBoundary[iter]) { firstEdge = iter; break; } MIntArray sortedEdges; bool isClosed = false; status = sortIntersections(itPolygon, itEdge, itVertex, sortedEdges, firstEdge, isClosed); if (1 < sortedEdges.length()) { // Create list of points from indices, avoid duplicates MPointArray sectionPoints; for (unsigned int i = 0; i < sortedEdges.length(); i++) if(0 == sectionPoints.length() || !sectionPoints[sectionPoints.length()-1].isEquivalent(m_intersectionPoints[sortedEdges[i]], 0.01)) sectionPoints.append(m_intersectionPoints[sortedEdges[i]]); // In case section should be closed but endpoints don't match, append first point to the end if (isClosed && sectionPoints[0] != sectionPoints[sectionPoints.length() - 1]) sectionPoints.append(sectionPoints[0]); MObject sectionCurve; generateNurbsCurve(sectionPoints, (isClosed) ? MFnNurbsCurve::kClosed : MFnNurbsCurve::kOpen, sectionCurve); sectionCurves.append(sectionCurve); } } return MS::kSuccess; } |
In every iteration we first add the intersection to a curve CV list and remove it from the intersection list. This signals the edge has been already iterated (to avoid getting stuck in an infinite loop).
Now we list the adjacent faces of this edge and go through all edges connected to these faces checking for any match in the intersection list. Once the intersection is found another sortIntersections function is called for the new edge. When no more connections can be found, the iteration cycle is ended and a new degree 1 curve is created from the CV list.
This would only work in case none of the intersections lied on a vertex. That’s the reason why we stored the intersection parameter in the first place. Whenever the parameter equals 0 or 1, we list the adjacent faces of the edge vertex, instead of adjacent faces of the edge. Also once the next intersection is found, all the edges connected to this vertext have to be removed from the intersection list. Here’s a little example to make everything a bit clearer (or confusing).
And here’s the sorting method in code. The 3 iterators – MItMeshPolygon, MItMeshEdge and MItMeshVertex – are passed as a parameter since initializing them inside the function every time was significantly decreasing performance.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 |
MStatus sortIntersections(MItMeshPolygon &itPolygon, MItMeshEdge &itEdge, MItMeshVertex &itVertex, MIntArray& sortedEdges, unsigned int currentEdge, bool &isClosed) { sortedEdges.append(currentEdge); m_intersectingEdges.erase(currentEdge); MIntArray connectedFaces; bool onBoundary; int dummyIndex; itEdge.setIndex(currentEdge, dummyIndex); onBoundary = itEdge.onBoundary(&status); // If inersection on vertex, get rid of faces sharing this vertex double parameter = m_intersectionParameters[currentEdge]; if (1==parameter || 0==parameter) { itVertex.setIndex(itEdge.index(int(parameter)), dummyIndex); itVertex.getConnectedFaces(connectedFaces); clearConnected(itVertex); } else itEdge.getConnectedFaces(connectedFaces, &status); // Iterate over connected faces, looking for next intersection for (unsigned int f = 0; f<connectedFaces.length(); f++) { itPolygon.setIndex(connectedFaces[f], dummyIndex); MIntArray connectedEdges; itPolygon.getEdges(connectedEdges); for (unsigned int e = 0; e < connectedEdges.length(); e++) if (m_intersectingEdges.end() != m_intersectingEdges.find(connectedEdges[e])) return sortIntersections(itPolygon, itEdge, itVertex, sortedEdges, connectedEdges[e], isClosed); } isClosed = (onBoundary) ? false : true; return MS::kSuccess; } |
Although in this case the sorting method deals with intersections between mesh and a plane, it can be used to sort intersection between mesh and any other geometry by simply changing the findIntersections function. Also don’t forget to check my github repository for the complete code.
No Comments
There are not comments on this post yet. Be the first one!