From 138a1d6f6f5deb0ef94d40d05ba0dd5ab1a122de Mon Sep 17 00:00:00 2001 From: Saleem Edah-Tally Date: Sat, 4 Jan 2025 21:15:22 +0100 Subject: [PATCH] Improve CenterlineDisassembly logic. CenterlineDisassembly - return the split centerline - handle an input centerline that does not have EdgeArray and/or EdgePCoordArray - do a NULL check - get number of branches, bifurcations and centerlines. --- .../CenterlineDisassembly.py | 99 ++++++++++++++----- 1 file changed, 74 insertions(+), 25 deletions(-) diff --git a/CenterlineDisassembly/CenterlineDisassembly.py b/CenterlineDisassembly/CenterlineDisassembly.py index 1c20732..43bde74 100644 --- a/CenterlineDisassembly/CenterlineDisassembly.py +++ b/CenterlineDisassembly/CenterlineDisassembly.py @@ -337,7 +337,7 @@ def splitCenterlines(self, inputCenterline: slicer.vtkMRMLModelNode): if not inputCenterline: raise ValueError(_("Input centerline is invalid")) - + import vtkvmtkComputationalGeometryPython as vtkvmtkComputationalGeometry branchExtractor = vtkvmtkComputationalGeometry.vtkvmtkCenterlineBranchExtractor() @@ -349,15 +349,53 @@ def splitCenterlines(self, inputCenterline: slicer.vtkMRMLModelNode): branchExtractor.SetTractIdsArrayName(tractIdsArrayName) branchExtractor.Update() self._splitCenterlines = branchExtractor.GetOutput() - + return self._splitCenterlines + + def getNumberOfCenterlines(self): + if not self._splitCenterlines: + raise ValueError(_("Call 'splitCenterlines()' with an input centerline model first.")) + + import vtkvmtkComputationalGeometryPython as vtkvmtkComputationalGeometry + centerlineIdsArray = self._splitCenterlines.GetCellData().GetArray(centerlineIdsArrayName) + centerlineIdsValueRange = centerlineIdsArray.GetValueRange() + # centerlineIdsValueRange[0] is always seen as 0. + return (centerlineIdsValueRange[1] - centerlineIdsValueRange[0]) + 1 + + def getNumberOfBifurcations(self): + # Logical bifurcations, not by anatomy. + if not self._splitCenterlines: + raise ValueError(_("Call 'splitCenterlines()' with an input centerline model first.")) + + groupIdsArray = vtk.vtkIdList() + import vtkvmtkComputationalGeometryPython as vtkvmtkComputationalGeometry + centerlineUtilities = vtkvmtkComputationalGeometry.vtkvmtkCenterlineUtilities() + centerlineUtilities.GetBlankedGroupsIdList(self._splitCenterlines, groupIdsArrayName, + blankingArrayName, groupIdsArray) + return groupIdsArray.GetNumberOfIds() + + def getNumberOfBranches(self): + # Logical branches, not by anatomy. + if not self._splitCenterlines: + raise ValueError(_("Call 'splitCenterlines()' with an input centerline model first.")) + + groupIdsArray = vtk.vtkIdList() + import vtkvmtkComputationalGeometryPython as vtkvmtkComputationalGeometry + centerlineUtilities = vtkvmtkComputationalGeometry.vtkvmtkCenterlineUtilities() + centerlineUtilities.GetNonBlankedGroupsIdList(self._splitCenterlines, groupIdsArrayName, + blankingArrayName, groupIdsArray) + return groupIdsArray.GetNumberOfIds() + def _createPolyData(self, cellIds): + if not self._splitCenterlines: + raise ValueError(_("Call 'splitCenterlines()' with an input centerline model first.")) + masterRadiusArray = self._splitCenterlines.GetPointData().GetArray(radiusArrayName) masterEdgeArray = self._splitCenterlines.GetPointData().GetArray(edgeArrayName) masterEdgePCoordArray = self._splitCenterlines.GetPointData().GetArray(edgePCoordArrayName) resultPolyDatas = [] # One per cell nbIds = cellIds.GetNumberOfIds() # Number of cells - + for i in range(nbIds): # For every cell unitCellPolyData = None pointId = 0 @@ -366,11 +404,13 @@ def _createPolyData(self, cellIds): cellArray = vtk.vtkCellArray() radiusArray = vtk.vtkDoubleArray() radiusArray.SetName(radiusArrayName) - edgeArray = vtk.vtkDoubleArray() - edgeArray.SetName("EdgeArray") - edgeArray.SetNumberOfComponents(2) - edgePCoordArray = vtk.vtkDoubleArray() - edgePCoordArray.SetName("EdgePCoordArray") + if masterEdgeArray: + edgeArray = vtk.vtkDoubleArray() + edgeArray.SetName("EdgeArray") + edgeArray.SetNumberOfComponents(2) + if masterEdgePCoordArray: + edgePCoordArray = vtk.vtkDoubleArray() + edgePCoordArray.SetName("EdgePCoordArray") masterCellId = cellIds.GetId(i) masterCellPolyLine = self._splitCenterlines.GetCell(masterCellId) @@ -384,9 +424,11 @@ def _createPolyData(self, cellIds): points.InsertNextPoint(point) cellArray.InsertCellPoint(pointId) radiusArray.InsertNextValue(masterRadiusArray.GetValue(masterPointId)) - edgeArray.InsertNextTuple2(masterEdgeArray.GetTuple2(masterPointId)[0], - masterEdgeArray.GetTuple2(masterPointId)[1]) - edgePCoordArray.InsertNextValue(masterEdgePCoordArray.GetValue(masterPointId)) + if masterEdgeArray: + edgeArray.InsertNextTuple2(masterEdgeArray.GetTuple2(masterPointId)[0], + masterEdgeArray.GetTuple2(masterPointId)[1]) + if masterEdgePCoordArray: + edgePCoordArray.InsertNextValue(masterEdgePCoordArray.GetValue(masterPointId)) pointId = pointId + 1 if (numberOfMasterCellPointIds): @@ -394,8 +436,10 @@ def _createPolyData(self, cellIds): unitCellPolyData.SetPoints(points) unitCellPolyData.SetLines(cellArray) unitCellPolyData.GetPointData().AddArray(radiusArray) - unitCellPolyData.GetPointData().AddArray(edgeArray) - unitCellPolyData.GetPointData().AddArray(edgePCoordArray) + if masterEdgeArray: + unitCellPolyData.GetPointData().AddArray(edgeArray) + if masterEdgePCoordArray: + unitCellPolyData.GetPointData().AddArray(edgePCoordArray) resultPolyDatas.append(unitCellPolyData) return resultPolyDatas @@ -442,11 +486,13 @@ def createCenterlineCurve(self, centerlinePolyData, curveNode): cellArray = vtk.vtkCellArray() radiusArray = vtk.vtkDoubleArray() radiusArray.SetName(radiusArrayName) - edgeArray = vtk.vtkDoubleArray() - edgeArray.SetName("EdgeArray") - edgeArray.SetNumberOfComponents(2) - edgePCoordArray = vtk.vtkDoubleArray() - edgePCoordArray.SetName("EdgePCoordArray") + if masterEdgeArray: + edgeArray = vtk.vtkDoubleArray() + edgeArray.SetName("EdgeArray") + edgeArray.SetNumberOfComponents(2) + if masterEdgePCoordArray: + edgePCoordArray = vtk.vtkDoubleArray() + edgePCoordArray.SetName("EdgePCoordArray") # The new cell array must allocate for points of all input cells. cellArray.InsertNextCell(centerlinePolyData.GetNumberOfPoints()) @@ -463,9 +509,11 @@ def createCenterlineCurve(self, centerlinePolyData, curveNode): points.InsertNextPoint(point) cellArray.InsertCellPoint(pointId) radiusArray.InsertNextValue(masterRadiusArray.GetValue(masterPointId)) - edgeArray.InsertNextTuple2(masterEdgeArray.GetTuple2(masterPointId)[0], - masterEdgeArray.GetTuple2(masterPointId)[1]) - edgePCoordArray.InsertNextValue(masterEdgePCoordArray.GetValue(masterPointId)) + if masterEdgeArray: + edgeArray.InsertNextTuple2(masterEdgeArray.GetTuple2(masterPointId)[0], + masterEdgeArray.GetTuple2(masterPointId)[1]) + if masterEdgePCoordArray: + edgePCoordArray.InsertNextValue(masterEdgePCoordArray.GetValue(masterPointId)) pointId = pointId + 1 # All cells from the input centerline have been processed. @@ -474,9 +522,10 @@ def createCenterlineCurve(self, centerlinePolyData, curveNode): newPolyData.SetPoints(points) newPolyData.SetLines(cellArray) newPolyData.GetPointData().AddArray(radiusArray) - newPolyData.GetPointData().AddArray(edgeArray) - newPolyData.GetPointData().AddArray(edgePCoordArray) - + if masterEdgeArray: + newPolyData.GetPointData().AddArray(edgeArray) + if masterEdgePCoordArray: + newPolyData.GetPointData().AddArray(edgePCoordArray) if curveNode and newPolyData: import ExtractCenterline @@ -530,8 +579,8 @@ def processGroupIds(self, bifurcations): logging.info(_("Processing group ids started")) groupIdsPolyDatas = [] - import vtkvmtkComputationalGeometryPython as vtkvmtkComputationalGeometry groupIdsArray = vtk.vtkIdList() + import vtkvmtkComputationalGeometryPython as vtkvmtkComputationalGeometry centerlineUtilities = vtkvmtkComputationalGeometry.vtkvmtkCenterlineUtilities() if (bifurcations): # Blanked