All Downloads are FREE. Search and download functionalities are using the official Maven repository.

scalismo.vtk.utils.Conversions.scala Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2015 University of Basel, Graphics and Vision Research Group
 *
 * 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.
 */
package scalismo.vtk.utils

import scalismo.common.{
  DiscreteField,
  PointId,
  PrimitiveScalarArray,
  Scalar,
  ScalarArray,
  ScalarMeshField3D,
  UnstructuredPoints,
  ValueClassScalarArray
}
import scalismo.common.DiscreteField.{ScalarMeshField, ScalarVolumeMeshField}
import scalismo.common.interpolation.BSplineImageInterpolator3D
import scalismo.geometry.{_2D, _3D, EuclideanVector, IntVector2D, IntVector3D, NDSpace, Point}
import scalismo.image.{DiscreteImage, DiscreteImageDomain, DiscreteImageDomain3D, StructuredPoints}
import scalismo.io.{ImageIO, ScalarDataType}
import scalismo.mesh.{
  LineCell,
  LineList,
  LineMesh,
  TetrahedralCell,
  TetrahedralList,
  TetrahedralMesh,
  TetrahedralMesh3D,
  TriangleCell,
  TriangleList,
  TriangleMesh,
  TriangleMesh3D
}
import scalismo.utils.ArrayUtils
import scalismo.utils.VtkHelpers.{
  VTK_CHAR,
  VTK_DOUBLE,
  VTK_FLOAT,
  VTK_INT,
  VTK_SHORT,
  VTK_SIGNED_CHAR,
  VTK_UNSIGNED_CHAR,
  VTK_UNSIGNED_INT,
  VTK_UNSIGNED_SHORT
}
import scalismo.vtk.utils.ImageConversion.{
  VtkAutomaticInterpolatorSelection,
  VtkCubicInterpolation,
  VtkInterpolationMode,
  VtkLinearInterpolation,
  VtkNearestNeighborInterpolation
}
import spire.math.{UByte, UInt, ULong, UShort}
import vtk.{
  vtkCellArray,
  vtkCharArray,
  vtkDataArray,
  vtkDoubleArray,
  vtkFloatArray,
  vtkIdList,
  vtkImageCast,
  vtkImageData,
  vtkImageReslice,
  vtkImageToStructuredPoints,
  vtkInformation,
  vtkIntArray,
  vtkLandmarkTransform,
  vtkLine,
  vtkLongArray,
  vtkPointSet,
  vtkPoints,
  vtkPolyData,
  vtkShortArray,
  vtkSignedCharArray,
  vtkStructuredPoints,
  vtkTetra,
  vtkTriangle,
  vtkTriangleFilter,
  vtkUnsignedCharArray,
  vtkUnsignedIntArray,
  vtkUnsignedShortArray,
  vtkUnstructuredGrid
}

import scala.reflect.ClassTag
import scala.util.{Failure, Success, Try}

object VtkHelpers {

  // ATTENTION: Writing out (signed) bytes using vtkCharArray seems to be broken in VTK, so we need to work around it.
  // We do this by writing the bytes into a vtkUnsignedCharArray first, then converting the scalar data.
  // This conversion must take place on the vtkStructuredPoints object containing the data, so we leave it to the caller of this method.
  def scalarArrayToVtkDataArray[A: Scalar](data: IndexedSeq[A], numComp: Int): vtkDataArray = {
    def init[T <: vtkDataArray](a: T): T = {
      a.SetNumberOfComponents(numComp)
      a.SetNumberOfTuples(data.length / numComp)
      a
    }

    Scalar[A].scalarType match {
      case Scalar.ShortScalar =>
        val a = init(new vtkShortArray())
        a.SetJavaArray(data.asInstanceOf[PrimitiveScalarArray[Short]].rawData)
        a
      case Scalar.IntScalar =>
        val a = init(new vtkIntArray())
        a.SetJavaArray(data.asInstanceOf[PrimitiveScalarArray[Int]].rawData)
        a
      case Scalar.LongScalar =>
        val a = init(new vtkLongArray())
        a.SetJavaArray(data.asInstanceOf[PrimitiveScalarArray[Long]].rawData)
        a
      case Scalar.FloatScalar =>
        val a = init(new vtkFloatArray())
        a.SetJavaArray(data.asInstanceOf[PrimitiveScalarArray[Float]].rawData)
        a
      case Scalar.DoubleScalar =>
        val a = init(new vtkDoubleArray())
        a.SetJavaArray(data.asInstanceOf[PrimitiveScalarArray[Double]].rawData)
        a
      case Scalar.ByteScalar =>
        /* ATTENTION: The following does NOT produce the correct result!
         * val a = init(new vtkCharArray())
         * a.SetJavaArray(data.asInstanceOf[Array[Byte]].map(_.toChar))
         * Here is the workaround:
         */
        val a = init(new vtkUnsignedCharArray())
        a.SetJavaArray(data.asInstanceOf[PrimitiveScalarArray[Byte]].rawData)
        a
      case Scalar.UByteScalar =>
        val a = init(new vtkUnsignedCharArray())
        //        a.SetJavaArray(data.asInstanceOf[Array[UByte]].map(_.toByte))
        a.SetJavaArray(data.asInstanceOf[ValueClassScalarArray[UByte, Byte]].rawData)
        a
      case Scalar.UShortScalar =>
        val a = init(new vtkUnsignedShortArray())
        val raw = data.asInstanceOf[ValueClassScalarArray[UShort, Char]].rawData
        a.SetJavaArray(ArrayUtils.fastMap[Char, Short](raw, _.toShort))
        a
      case Scalar.UIntScalar =>
        val a = init(new vtkUnsignedIntArray())
        a.SetJavaArray(data.asInstanceOf[ValueClassScalarArray[UInt, Int]].rawData)
        a
    }
  }

  def vtkDataArrayToScalarArray[A: ClassTag: Scalar](vtkType: Int, arrayVTK: vtkDataArray): Try[ScalarArray[A]] = Try {
    val scalar = Scalar[A]

    // There seems to be a bug in VTK, thathe array 1 element to big. To work
    // around this, we compute the number of elements manually and take only this number of elements.
    val numElementsInArray = arrayVTK.GetNumberOfTuples().toInt * arrayVTK.GetNumberOfComponents()

    vtkType match {
      // simple cases, no magic needed
      case VTK_SHORT =>
        val p = arrayVTK.asInstanceOf[vtkShortArray].GetJavaArray().take(numElementsInArray)
        Scalar.ShortIsScalar.createArray(p).map((s: Short) => scalar.fromShort(s))
      case VTK_INT =>
        val p = arrayVTK.asInstanceOf[vtkIntArray].GetJavaArray().take(numElementsInArray)
        Scalar.IntIsScalar.createArray(p).map((i: Int) => scalar.fromInt(i))
      case VTK_FLOAT =>
        val p = arrayVTK.asInstanceOf[vtkFloatArray].GetJavaArray().take(numElementsInArray)
        Scalar.FloatIsScalar.createArray(p).map((f: Float) => scalar.fromFloat(f))
      case VTK_DOUBLE =>
        val p = arrayVTK.asInstanceOf[vtkDoubleArray].GetJavaArray().take(numElementsInArray)
        Scalar.DoubleIsScalar.createArray(p).map((d: Double) => scalar.fromDouble(d))
      // complicated cases, so we're more explicit about what we're doing
      case VTK_CHAR =>
        val in = arrayVTK.asInstanceOf[vtkCharArray] // .GetJavaArray()
        val out: Array[Byte] = new Array[Byte](numElementsInArray)
        var i = 0
        while (i < out.length) {
          out(i) = in.GetValue(i).toByte
          i += 1
        }
        Scalar.ByteIsScalar.createArray(out).map((b: Byte) => scalar.fromByte(b))
      case VTK_SIGNED_CHAR =>
        val in = arrayVTK.asInstanceOf[vtkSignedCharArray]

        // vtkSignedCharArray does not seem to have a GetJavaArray method.
        // We therefore need to copy it manually
        val out: Array[Byte] = new Array[Byte](numElementsInArray)
        var i = 0
        while (i < out.length) {
          out(i) = in.GetValue(i).toByte
          i += 1
        }
        Scalar.ByteIsScalar.createArray(out).map((b: Byte) => scalar.fromByte(b))
      case VTK_UNSIGNED_CHAR =>
        val in = arrayVTK.asInstanceOf[vtkUnsignedCharArray].GetJavaArray().take(numElementsInArray)
        Scalar.UByteIsScalar.createArray(in).map((s: UByte) => scalar.fromShort(s.toShort))
      case VTK_UNSIGNED_SHORT =>
        val in = arrayVTK.asInstanceOf[vtkUnsignedShortArray].GetJavaArray().take(numElementsInArray)
        val chars = ArrayUtils.fastMap[Short, Char](in, _.toChar)
        Scalar.UShortIsScalar.createArray(chars).map((s: UShort) => scalar.fromInt(s.toInt))
      case VTK_UNSIGNED_INT =>
        val in = arrayVTK.asInstanceOf[vtkUnsignedIntArray].GetJavaArray().take(numElementsInArray)
        Scalar.UIntIsScalar.createArray(in).asInstanceOf[ScalarArray[A]]
      case _ => throw new NotImplementedError("Unsupported Scalar Pixel Type " + Scalar[A].scalarType)
    }
  }

}

object TetrahedralMeshConversion {

  private def extractPointsAndCells(ug: vtkUnstructuredGrid) = Try {
    val grids = ug.GetCells()
    val numGrids = grids.GetNumberOfCells().toInt

    val points = CommonConversions.vtkConvertPoints[_3D](ug)

    val idList = new vtkIdList()

    val cells = for (i <- 0 until numGrids) yield {
      ug.GetCellPoints(i, idList)
      if (idList.GetNumberOfIds() != 4) {
        throw new Exception("Not a tetrahedral mesh")
      }

      TetrahedralCell(PointId(idList.GetId(0).toInt),
                      PointId(idList.GetId(1).toInt),
                      PointId(idList.GetId(2).toInt),
                      PointId(idList.GetId(3).toInt)
      )
    }
    idList.Delete()
    (points, cells)
  }

  def vtkUnstructuredGridToTetrahedralMesh(ug: vtkUnstructuredGrid): Try[TetrahedralMesh[_3D]] = {
    val cellsPointsOrFailure = extractPointsAndCells(ug)
    cellsPointsOrFailure.map { case (points, cells) =>
      TetrahedralMesh3D(UnstructuredPoints(points.toIndexedSeq), TetrahedralList(cells))
    }
  }

  def tetrahedralMeshToVTKUnstructuredGrid(tetramesh: TetrahedralMesh[_3D],
                                           template: Option[vtkUnstructuredGrid] = None
  ): vtkUnstructuredGrid = {

    val ug = new vtkUnstructuredGrid()

    template match {
      case Some(tpl) => {
        // copy tetrahedrons from template if given; actual points are set unconditionally in code below.
        ug.ShallowCopy(tpl)
      }
      case None => {
        val VTK_TETRA = new vtk.vtkTetra().GetCellType()
        val tetrahedrons = new vtkCellArray
        tetrahedrons.SetNumberOfCells(tetramesh.tetrahedralization.tetrahedrons.size)
        tetrahedrons.Initialize()

        for ((cell, cell_id) <- tetramesh.tetrahedralization.tetrahedrons.zipWithIndex) {
          val tetrahedron = new vtkTetra()
          tetrahedron.GetPointIds().SetId(0, cell.ptId1.id)
          tetrahedron.GetPointIds().SetId(1, cell.ptId2.id)
          tetrahedron.GetPointIds().SetId(2, cell.ptId3.id)
          tetrahedron.GetPointIds().SetId(3, cell.ptId4.id)

          tetrahedrons.InsertNextCell(tetrahedron)
        }

        tetrahedrons.Squeeze()
        ug.SetCells(VTK_TETRA, tetrahedrons)
      }
    }

    // set points
    val pointDataArray = tetramesh.pointSet.pointSequence.toArray.flatMap(_.toArray)
    val pointDataArrayVTK = VtkHelpers.scalarArrayToVtkDataArray(Scalar.DoubleIsScalar.createArray(pointDataArray), 3)
    val pointsVTK = new vtkPoints
    pointsVTK.SetData(pointDataArrayVTK)
    ug.SetPoints(pointsVTK)
    ug
  }

  def scalarVolumeMeshFieldToVtkUnstructuredGrid[S: Scalar: ClassTag](
    tetraMeshData: ScalarVolumeMeshField[S],
    template: Option[vtkUnstructuredGrid] = None
  ): vtkUnstructuredGrid = {

    val grid = tetrahedralMeshToVTKUnstructuredGrid(tetraMeshData.mesh, template)
    val scalarData = VtkHelpers.scalarArrayToVtkDataArray(ScalarArray(tetraMeshData.data.toArray), 1)
    grid.GetPointData().SetScalars(scalarData)
    grid
  }

  def vtkUnstructuredGridToScalarVolumeMeshField[S: Scalar: ClassTag](
    ug: vtkUnstructuredGrid
  ): Try[ScalarVolumeMeshField[S]] = {
    for {
      mesh <- vtkUnstructuredGridToTetrahedralMesh(ug)
      scalarData <- VtkHelpers
        .vtkDataArrayToScalarArray[S](ug.GetPointData().GetScalars().GetDataType(), ug.GetPointData().GetScalars())
    } yield {
      DiscreteField(mesh, scalarData)
    }
  }

}

object MeshConversion {

  private def vtkPolyDataToTriangleMeshCommon(pd: vtkPolyData, correctFlag: Boolean = false) = Try {

    val newPd = if (correctFlag) {
      val triangleFilter = new vtkTriangleFilter
      triangleFilter.SetInputData(pd)
      triangleFilter.Update()
      triangleFilter.GetOutput()
    } else pd

    val polys = newPd.GetPolys()
    val numPolys = polys.GetNumberOfCells().toInt

    val points = CommonConversions.vtkConvertPoints[_3D](newPd)

    val idList = new vtkIdList()
    val cells = for (i <- 0 until numPolys) yield {
      newPd.GetCellPoints(i, idList)
      if (idList.GetNumberOfIds() != 3) {
        throw new Exception("Not a triangle mesh")
      }

      TriangleCell(PointId(idList.GetId(0).toInt), PointId(idList.GetId(1).toInt), PointId(idList.GetId(2).toInt))
    }
    idList.Delete()
    (points, cells)
  }

  def vtkPolyDataToTriangleMesh(pd: vtkPolyData): Try[TriangleMesh[_3D]] = {
    // TODO currently all data arrays are ignored
    val cellsPointsOrFailure = vtkPolyDataToTriangleMeshCommon(pd)
    cellsPointsOrFailure.map { case (points, cells) =>
      TriangleMesh3D(UnstructuredPoints(points.toIndexedSeq), TriangleList(cells))
    }
  }

  def vtkPolyDataToCorrectedTriangleMesh(pd: vtkPolyData): Try[TriangleMesh[_3D]] = {

    val cellsPointsOrFailure = vtkPolyDataToTriangleMeshCommon(pd, correctFlag = true)
    cellsPointsOrFailure.map { case (points, cells) =>
      val cellPointIds = cells.flatMap(_.pointIds).distinct
      val oldId2newId = cellPointIds.zipWithIndex.map { case (id, index) => (id, PointId(index)) }.toMap
      val newCells = cells.map(c => TriangleCell(oldId2newId(c.ptId1), oldId2newId(c.ptId2), oldId2newId(c.ptId3)))
      val oldPoints = points.toIndexedSeq
      val newPoints = cellPointIds.map(id => oldPoints(id.id))
      TriangleMesh3D(UnstructuredPoints(newPoints), TriangleList(newCells))
    }
  }

  def meshToVtkPolyData(mesh: TriangleMesh[_3D], template: Option[vtkPolyData] = None): vtkPolyData = {

    val pd = new vtkPolyData

    template match {
      case Some(tpl) =>
        // copy triangles from template if given; actual points are set unconditionally in code below.
        pd.ShallowCopy(tpl)
      case None =>
        val triangles = new vtkCellArray
        triangles.SetNumberOfCells(mesh.triangulation.triangles.size)
        triangles.Initialize()
        for ((cell, cell_id) <- mesh.triangulation.triangles.zipWithIndex) {
          val triangle = new vtkTriangle()

          triangle.GetPointIds().SetId(0, cell.ptId1.id)
          triangle.GetPointIds().SetId(1, cell.ptId2.id)
          triangle.GetPointIds().SetId(2, cell.ptId3.id)
          triangles.InsertNextCell(triangle)
        }
        triangles.Squeeze()
        pd.SetPolys(triangles)
    }

    // set points
    val pointDataArray = mesh.pointSet.pointSequence.toArray.flatMap(_.toArray)
    val pointDataArrayVTK = VtkHelpers.scalarArrayToVtkDataArray(Scalar.DoubleIsScalar.createArray(pointDataArray), 3)
    val pointsVTK = new vtkPoints
    pointsVTK.SetData(pointDataArrayVTK)
    pd.SetPoints(pointsVTK)

    pd
  }

  def scalarMeshFieldToVtkPolyData[S: Scalar: ClassTag](meshData: ScalarMeshField[S],
                                                        template: Option[vtkPolyData] = None
  ): vtkPolyData = {
    val pd = meshToVtkPolyData(meshData.mesh, template)
    val scalarData =
      VtkHelpers.scalarArrayToVtkDataArray(ScalarArray(meshData.data.toArray), 1) // TODO make this more general
    pd.GetPointData().SetScalars(scalarData)
    pd
  }

  def vtkPolyDataToScalarMeshField[S: Scalar: ClassTag](pd: vtkPolyData): Try[ScalarMeshField[S]] = {
    for {
      mesh <- vtkPolyDataToTriangleMesh(pd)
      scalarData <- VtkHelpers
        .vtkDataArrayToScalarArray[S](pd.GetPointData().GetScalars().GetDataType(), pd.GetPointData().GetScalars())
    } yield {
      ScalarMeshField3D(mesh, scalarData)
    }
  }

  def vtkPolyDataToLineMesh[D: NDSpace: LineMesh.Create: UnstructuredPoints.Create](
    pd: vtkPolyData
  ): Try[LineMesh[D]] = {
    val lines = pd.GetLines()
    val numPolys = lines.GetNumberOfCells().toInt
    val points = CommonConversions.vtkConvertPoints[D](pd)
    val idList = new vtkIdList()
    val cellsOrFailure = Try {
      for (i <- 0 until numPolys) yield {
        pd.GetCellPoints(i, idList)
        if (idList.GetNumberOfIds() != 2) {
          throw new scala.Exception("Not a poly line")
        } else {
          LineCell(PointId(idList.GetId(0).toInt), PointId(idList.GetId(1).toInt))
        }
      }
    }
    idList.Delete()

    cellsOrFailure.map(cells => LineMesh(UnstructuredPoints[D](points.toIndexedSeq), LineList(cells)))
  }

  def lineMeshToVTKPolyData[D: NDSpace](mesh: LineMesh[D], template: Option[vtkPolyData] = None): vtkPolyData = {

    val pd = new vtkPolyData

    template match {
      case Some(tpl) =>
        // copy triangles from template if given; actual points are set unconditionally in code below.
        pd.ShallowCopy(tpl)
      case None =>
        val lines = new vtkCellArray
        lines.SetNumberOfCells(mesh.lines.size)
        lines.Initialize()
        for ((cell, cell_id) <- mesh.lines.zipWithIndex) {
          val line = new vtkLine()

          line.GetPointIds().SetId(0, cell.ptId1.id)
          line.GetPointIds().SetId(1, cell.ptId2.id)
          lines.InsertNextCell(line)
        }
        lines.Squeeze()
        pd.SetLines(lines)
    }

    // set points. As vtk knows only about 3D poitns, we lift our 2D points to 3D-
    val pointDim = NDSpace[D].dimensionality
    val meshPointsIn3D = if (pointDim == 2) {
      mesh.pointSet.points.map(p => Point(p(0), p(1), 0f))
    } else {
      mesh.pointSet.points.map(p => Point(p(0), p(1), p(2)))
    }
    val pointDataArray = meshPointsIn3D.toIndexedSeq.toArray.flatMap(_.toArray).map(_.toFloat)
    val pointDataArrayVTK = VtkHelpers.scalarArrayToVtkDataArray(Scalar.FloatIsScalar.createArray(pointDataArray), 3)
    val pointsVTK = new vtkPoints
    pointsVTK.SetData(pointDataArrayVTK)
    pd.SetPoints(pointsVTK)

    pd
  }

}

object CommonConversions {

  private[scalismo] def vtkConvertPoints[D: NDSpace](pd: vtkPointSet): Iterator[Point[D]] = {
    val vtkType = pd.GetPoints().GetDataType()

    val pointsArray = VtkHelpers.vtkDataArrayToScalarArray[Float](vtkType, pd.GetPoints().GetData()) match {
      case Success(data) => data.toArray
      // this should actually never happen. The points array can only be float or double, and hence
      // vtkDataArrayToScalarArray should always return success. If not, something very strange is happening,
      // in which case we just throw the exception
      case Failure(t) => throw t
    }

    // vtk point are alwyas 3D. Therefore we take all three coordinates out of the array but,
    // if we are in 2D, take only the first 2. Finally, we need to convert them from float to double.
    pointsArray
      .grouped(3)
      .map(p => Point(p.take(NDSpace[D].dimensionality).map(_.toDouble)))
  }

}

trait CanConvertToVtk[D] {

  def toVtk[Pixel: Scalar: ClassTag](img: DiscreteImage[D, Pixel],
                                     interpolationMode: VtkInterpolationMode
  ): vtkStructuredPoints = {

    val sp = new vtkStructuredPoints()
    sp.SetNumberOfScalarComponents(1, new vtkInformation())
    val dataArray = img.data match {
      case data: ScalarArray[Pixel @unchecked] => VtkHelpers.scalarArrayToVtkDataArray(data, 1)
      case data: IndexedSeq[Pixel @unchecked]  => VtkHelpers.scalarArrayToVtkDataArray(ScalarArray(data.toArray), 1)
    }
    sp.GetPointData().SetScalars(dataArray)

    // In the case of 3D, this might create a new vtkStructuredPoints data due to image orientation
    val orientedSP = setDomainInfo(img.domain.pointSet, sp, interpolationMode)

    if (Scalar[Pixel].scalarType == Scalar.ByteScalar) {
      recastDataToSignedChar(orientedSP)
    }

    orientedSP
  }

  def setDomainInfo(domain: StructuredPoints[D],
                    sp: vtkStructuredPoints,
                    interpolationMode: VtkInterpolationMode
  ): vtkStructuredPoints

  def fromVtk[Pixel: Scalar: ClassTag](sp: vtkImageData): Try[DiscreteImage[D, Pixel]]

  // ATTENTION: Writing out (signed) bytes using vtkCharArray seems to be broken in VTK, so we need to work around it.
  // We do this by writing the bytes into a vtkUnsignedCharArray first, then converting the scalar data.
  // Also see VtkHelpers.javaArrayToVtkDataArray().
  private def recastDataToSignedChar(sp: vtkStructuredPoints): Unit = {
    val cast = new vtkImageCast
    cast.SetInputData(sp)
    cast.SetOutputScalarTypeToChar()
    cast.Update()
    val csp = cast.GetOutput()
    sp.GetPointData().GetScalars().Delete()
    sp.GetPointData().SetScalars(csp.GetPointData().GetScalars())
    cast.Delete()
    csp.Delete()
  }
}

object CanConvertToVtk {

  implicit object _2DCanConvertToVtk$ extends CanConvertToVtk[_2D] {

    override def setDomainInfo(domain: StructuredPoints[_2D],
                               sp: vtkStructuredPoints,
                               interpolationMode: VtkInterpolationMode
    ): vtkStructuredPoints = {
      sp.SetDimensions(domain.size(0), domain.size(1), 1)
      sp.SetOrigin(domain.origin(0), domain.origin(1), 0)
      sp.SetSpacing(domain.spacing(0), domain.spacing(1), 0)
      sp
    }

    override def fromVtk[Pixel: Scalar: ClassTag](sp: vtkImageData): Try[DiscreteImage[_2D, Pixel]] = {
      if (sp.GetNumberOfScalarComponents() != 1) {
        return Failure(
          new Exception(s"The image is not a scalar image (number of components is ${sp.GetNumberOfScalarComponents()}")
        )
      }

      if (sp.GetDimensions()(2) != 1 && sp.GetDimensions()(1) != 0) {
        return Failure(new Exception(s"The image is a 3D image - require a 2D image"))
      }

      val requiredScalarType = ScalarDataType.fromType[Pixel]
      val spScalarType = ScalarDataType.fromVtkId(sp.GetScalarType())

      if (requiredScalarType != spScalarType) {
        return Failure(new Exception(s"Invalid scalar type (expected $requiredScalarType, found $spScalarType)"))
      }

      val origin = Point(sp.GetOrigin()(0).toFloat, sp.GetOrigin()(1).toFloat)
      val spacing = EuclideanVector(sp.GetSpacing()(0).toFloat, sp.GetSpacing()(1).toFloat)
      val size = IntVector2D(sp.GetDimensions()(0), sp.GetDimensions()(1))

      val domain = DiscreteImageDomain(StructuredPoints[_2D](origin, spacing, size))
      val scalars = sp.GetPointData().GetScalars()

      val pixelArrayOrFailure = VtkHelpers.vtkDataArrayToScalarArray[Pixel](sp.GetScalarType(), scalars)
      pixelArrayOrFailure.map(pixelArray => DiscreteImage(domain, pixelArray))

    }

  }

  implicit object _3DCanConvertToVtk$ extends CanConvertToVtk[_3D] {
    override def setDomainInfo(domain: StructuredPoints[_3D],
                               sp: vtkStructuredPoints,
                               interpolationMode: VtkInterpolationMode
    ): vtkStructuredPoints = {

      // Here depending on the image directions (if read from Nifti, can be anything RAS, ASL, LAS, ..),
      // we need to reslice the image in vtk's voxel ordering that is RAI (which in our LPS world coordinates system
      // means +i <=> +x, +j<=>+y, +k<=>+z)

      sp.SetDimensions(domain.size(0), domain.size(1), domain.size(2))

      val corners = List(
        IntVector3D(0, 0, 0),
        IntVector3D(domain.size(0) - 1, 0, 0),
        IntVector3D(0, domain.size(1) - 1, 0),
        IntVector3D(0, 0, domain.size(2) - 1),
        IntVector3D(domain.size(0) - 1, domain.size(1) - 1, 0),
        IntVector3D(domain.size(0) - 1, 0, domain.size(2) - 1),
        IntVector3D(0, domain.size(1) - 1, domain.size(2) - 1),
        IntVector3D(domain.size(0) - 1, domain.size(1) - 1, domain.size(2) - 1)
      )
      val cornerImages = corners.map(domain.indexToPoint)
      val newOriginX = cornerImages.map(p => p.x).min
      val newOriginY = cornerImages.map(p => p.y).min
      val newOriginZ = cornerImages.map(p => p.z).min

      val vtkSourceCorners = new vtkPoints()
      corners.foreach(c => vtkSourceCorners.InsertNextPoint(c.toArray.map(_.toDouble)))

      val vtkTargetCorners = new vtkPoints()
      cornerImages.foreach(c => vtkTargetCorners.InsertNextPoint(c.toArray.map(_.toDouble)))

      val landmarkTransform = new vtkLandmarkTransform()

      landmarkTransform.SetSourceLandmarks(vtkTargetCorners)
      landmarkTransform.SetTargetLandmarks(vtkSourceCorners)
      landmarkTransform.SetModeToAffine()
      landmarkTransform.Update()

      val reslice = new vtkImageReslice()

      reslice.SetInputData(sp)
      reslice.SetResliceTransform(landmarkTransform)

      interpolationMode match {
        case VtkCubicInterpolation             => reslice.SetInterpolationModeToCubic()
        case VtkLinearInterpolation            => reslice.SetInterpolationModeToLinear()
        case VtkNearestNeighborInterpolation   => reslice.SetInterpolationModeToNearestNeighbor()
        case VtkAutomaticInterpolatorSelection => {}
      }

      reslice.SetOutputSpacing(domain.spacing(0), domain.spacing(1), domain.spacing(2))
      reslice.SetOutputOrigin(newOriginX, newOriginY, newOriginZ)

      val newXSpatialSize = cornerImages.map(p => p.x).max - newOriginX
      val newYSpatialSize = cornerImages.map(p => p.y).max - newOriginY
      val newZSpatialSize = cornerImages.map(p => p.z).max - newOriginZ

      val newXExtent = math.round(newXSpatialSize / domain.spacing(0)).toInt
      val newYExtent = math.round(newYSpatialSize / domain.spacing(1)).toInt
      val newZExtent = math.round(newZSpatialSize / domain.spacing(2)).toInt

      reslice.SetOutputExtent(0, newXExtent, 0, newYExtent, 0, newZExtent)

      val conv = new vtkImageToStructuredPoints()
      conv.SetInputConnection(reslice.GetOutputPort())
      conv.Update()
      conv.GetStructuredPointsOutput()
    }

    override def fromVtk[Pixel: Scalar: ClassTag](sp: vtkImageData): Try[DiscreteImage[_3D, Pixel]] = {
      if (sp.GetNumberOfScalarComponents() != 1) {
        return Failure(
          new Exception(s"The image is not a scalar image (number of components is ${sp.GetNumberOfScalarComponents()}")
        )
      }

      if (sp.GetDimensions()(2) == 1 || sp.GetDimensions()(2) == 0) {
        return Failure(new Exception(s"The image is a 2D image - require a 3D image"))
      }

      val requiredScalarType = ScalarDataType.fromType[Pixel]
      val spScalarType = ScalarDataType.fromVtkId(sp.GetScalarType())

      if (requiredScalarType != spScalarType) {
        return Failure(new Exception(s"Invalid scalar type (expected $requiredScalarType, found $spScalarType)"))
      }

      val origin = Point(sp.GetOrigin()(0).toFloat, sp.GetOrigin()(1).toFloat, sp.GetOrigin()(2).toFloat)
      val spacing = EuclideanVector(sp.GetSpacing()(0).toFloat, sp.GetSpacing()(1).toFloat, sp.GetSpacing()(2).toFloat)
      val size = IntVector3D(sp.GetDimensions()(0), sp.GetDimensions()(1), sp.GetDimensions()(2))

      val domain = DiscreteImageDomain(StructuredPoints[_3D](origin, spacing, size))
      val scalars = sp.GetPointData().GetScalars()
      val pixelArrayOrFailure = VtkHelpers.vtkDataArrayToScalarArray[Pixel](sp.GetScalarType(), scalars)
      pixelArrayOrFailure.map(pixelArray => DiscreteImage(domain, pixelArray))
    }
  }

}

object ImageConversion {

  sealed trait VtkInterpolationMode
  case object VtkCubicInterpolation extends VtkInterpolationMode
  case object VtkNearestNeighborInterpolation extends VtkInterpolationMode
  case object VtkLinearInterpolation extends VtkInterpolationMode
  case object VtkAutomaticInterpolatorSelection extends VtkInterpolationMode

  def imageToVtkStructuredPoints[D: CanConvertToVtk, Pixel: Scalar: ClassTag](
    img: DiscreteImage[D, Pixel],
    interpolationMode: VtkInterpolationMode = VtkAutomaticInterpolatorSelection
  ): vtkStructuredPoints = {

    // If the interpolation model is set to automatic, we apply
    //  a crude heuristic to understand how we need to interpolate. If the number of values is small
    // we are likely dealing with a labelmap and should interpolate using Nearest Neighbor.
    // If the number of values is large, we are dealing with a normal image
    // and should use higher order interpolation schemes.
    val selectedInterpolationMode: VtkInterpolationMode = interpolationMode match {
      case VtkAutomaticInterpolatorSelection =>
        if (img.values.toSet.size <= 255) {
          ImageConversion.VtkNearestNeighborInterpolation
        } else {
          ImageConversion.VtkCubicInterpolation
        }
      case _ => interpolationMode
    }
    implicitly[CanConvertToVtk[D]].toVtk(img, selectedInterpolationMode)

  }

  def vtkStructuredPointsToScalarImage[D: CanConvertToVtk, Pixel: Scalar: ClassTag](
    sp: vtkImageData
  ): Try[DiscreteImage[D, Pixel]] = {
    implicitly[CanConvertToVtk[D]].fromVtk(sp)
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy