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

scalismo.vtk.io.MeshIO.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.io

import scalismo.color.{RGB, RGBA}
import scalismo.common.DiscreteField.{ScalarMeshField, ScalarVolumeMeshField}
import scalismo.common.{PointId, Scalar, UnstructuredPoints}
import scalismo.geometry.*
import scalismo.hdf5json.HDFPath
import scalismo.io.ScalarDataType
import scalismo.io.statisticalmodel.{NDArray, StatisticalModelIOUtils}
import scalismo.mesh.*
import scalismo.mesh.TriangleMesh.*
import scalismo.vtk.utils.{MeshConversion, TetrahedralMeshConversion}
import vtk.*

import java.io.{BufferedReader, File, FileReader, IOException}
import scala.reflect.ClassTag
import scala.util.{Failure, Success, Try}

object MeshIO {

  /**
   * Implements methods for reading and writing D-dimensional meshes
   *
   * '''WARNING! WE ARE USING an LPS WORLD COORDINATE SYSTEM'''
   *
   * This means that when reading mesh files such as .stl or .vtk, we assume the point coordinates to lie in an LPS
   * world and map them unchanged in our coordinate system.
   *
   * The same happens at writing, we directly dump our vertex coordinates into the file format(stl, or vtk) without any
   * mirroring magic.
   *
   * *
   */
  /**
   * Reads a ScalarMeshField from file. The indicated Scalar type S must match the data type encoded in the file
   */
  def readScalarMeshField[S: Scalar: ClassTag](file: File): Try[ScalarMeshField[S]] = {
    val requiredScalarType = ScalarDataType.fromType[S]
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") =>
        readVTKPolydata(file).flatMap { pd =>
          val spScalarType = ScalarDataType.fromVtkId(pd.GetPointData().GetScalars().GetDataType())
          MeshConversion.vtkPolyDataToScalarMeshField(pd)
          if (requiredScalarType != spScalarType) {
            Failure(new Exception(s"Invalid scalar type (expected $requiredScalarType, found $spScalarType)"))
          } else {
            MeshConversion.vtkPolyDataToScalarMeshField(pd)
          }
        }
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  /**
   * Reads a ScalarMeshField from file while casting its data to the indicated Scalar type S if necessary
   */
  def readScalarMeshFieldAsType[S: Scalar: ClassTag](file: File): Try[ScalarMeshField[S]] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") =>
        readVTKPolydata(file).flatMap(pd => MeshConversion.vtkPolyDataToScalarMeshField(pd))
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def readMesh(file: File): Try[TriangleMesh[_3D]] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") => readVTK(file)
      case f if f.endsWith(".stl") => readSTL(file)
      case f if f.endsWith(".ply") => {
        readPLY(file).map { res =>
          res match {
            case Right(vertexColor) => vertexColor.shape
            case Left(shape)        => shape
          }
        }
      }
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def readVertexColorMesh3D(file: File): Try[VertexColorMesh3D] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".ply") =>
        readPLY(file).map { r =>
          r match {
            case Right(colorMesh3D) => colorMesh3D
            case Left(_)            => throw new Exception("Indicated PLY file does not contain color values.")
          }
        }

      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def readAndCorrectMesh(file: File): Try[TriangleMesh[_3D]] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") => readVTK(file, correctMesh = true)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def readLineMesh2D(file: File): Try[LineMesh[_2D]] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") => readLineMeshVTK(file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }

  }

  def readLineMesh3D(file: File): Try[LineMesh[_3D]] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") => readLineMeshVTK(file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def writeLineMesh[D: NDSpace](polyLine: LineMesh[D], file: File): Try[Unit] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") => writeLineMeshVTK(polyLine, file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  /**
   * Reads a [[TetrahedralMesh[_3D]]] from a file with one of the extensions ".vtk", ".vtu", or ".inp". The ".vtk" and
   * ".vtu" files are standard VTK formats while ".inp" is the AVS UCD format.
   */
  def readTetrahedralMesh(file: File): Try[TetrahedralMesh[_3D]] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".inp") => readFromVTKFileThenDelete(readVTKAVSucdUnstructuredGrid, file)
      case f if f.endsWith(".vtk") => readFromVTKFileThenDelete(readVTKUnstructuredGrid, file)
      case f if f.endsWith(".vtu") => readFromVTKFileThenDelete(readVTKXMLUnstructuredGrid, file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def readScalarVolumeMeshField[S: Scalar: ClassTag](file: File): Try[ScalarVolumeMeshField[S]] = {
    val requiredScalarType = ScalarDataType.fromType[S]
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") =>
        readVTKUnstructuredGrid(file).flatMap { ug =>
          val spScalarType = ScalarDataType.fromVtkId(ug.GetPointData().GetScalars().GetDataType())
          TetrahedralMeshConversion.vtkUnstructuredGridToScalarVolumeMeshField(ug)
          if (requiredScalarType != spScalarType) {
            Failure(new Exception(s"Invalid scalar type (expected $requiredScalarType, found $spScalarType)"))
          } else {
            TetrahedralMeshConversion.vtkUnstructuredGridToScalarVolumeMeshField(ug)
          }
        }
      case f if f.endsWith(".vtu") =>
        readVTKXMLUnstructuredGrid(file).flatMap { ug =>
          val spScalarType = ScalarDataType.fromVtkId(ug.GetPointData().GetScalars().GetDataType())
          TetrahedralMeshConversion.vtkUnstructuredGridToScalarVolumeMeshField(ug)
          if (requiredScalarType != spScalarType) {
            Failure(new Exception(s"Invalid scalar type (expected $requiredScalarType, found $spScalarType)"))
          } else {
            TetrahedralMeshConversion.vtkUnstructuredGridToScalarVolumeMeshField(ug)
          }
        }
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }
  def readScalarVolumeMeshFieldAsType[S: Scalar: ClassTag](file: File): Try[ScalarVolumeMeshField[S]] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") =>
        readVTKUnstructuredGrid(file).flatMap(ug =>
          TetrahedralMeshConversion.vtkUnstructuredGridToScalarVolumeMeshField(ug)
        )
      case f if f.endsWith(".vtu") =>
        readVTKXMLUnstructuredGrid(file).flatMap(ug =>
          TetrahedralMeshConversion.vtkUnstructuredGridToScalarVolumeMeshField(ug)
        )
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  private[io] def readFromVTKFileThenDelete(readUSFromFile: File => Try[vtkUnstructuredGrid],
                                            file: File
  ): Try[TetrahedralMesh[_3D]] = {
    for {
      vtkUg <- readUSFromFile(file)
      tetramesh <- TetrahedralMeshConversion.vtkUnstructuredGridToTetrahedralMesh(vtkUg)
    } yield {
      vtkUg.Delete()
      tetramesh
    }
  }

  private[io] def readFromVTKFileThenDelete[S: Scalar: ClassTag](
    readUSFromFile: File => Try[vtkUnstructuredGrid],
    file: File
  ): Try[ScalarVolumeMeshField[S]] = {
    for {
      vtkUg <- readUSFromFile(file)
      tetrafield <- TetrahedralMeshConversion.vtkUnstructuredGridToScalarVolumeMeshField(vtkUg)
    } yield {
      vtkUg.Delete()
      tetrafield
    }
  }

  private[io] def readVTKUnstructuredGrid(file: File): Try[vtkUnstructuredGrid] = {

    val vtkReader = new vtkUnstructuredGridReader()
    vtkReader.SetFileName(file.getAbsolutePath)
    vtkReader.Update()

    val extract = new vtkExtractUnstructuredGrid()
    extract.SetInputConnection(vtkReader.GetOutputPort())

    val errCode = vtkReader.GetErrorCode()
    if (errCode != 0) {
      return Failure(new IOException(s"Could not read vtk UnstructuredGrid (received error code $errCode"))
    }
    val data = vtkReader.GetOutput()
    vtkReader.Delete()
    Success(data)
  }

  private[io] def readVTKXMLUnstructuredGrid(file: File): Try[vtkUnstructuredGrid] = {

    val vtkReader = new vtkXMLUnstructuredGridReader()
    vtkReader.SetFileName(file.getAbsolutePath)
    vtkReader.Update()

    val errCode = vtkReader.GetErrorCode()
    if (errCode != 0) {
      return Failure(new IOException(s"Could not read vtk UnstructuredGrid (received error code $errCode"))
    }
    val data = vtkReader.GetOutput()
    vtkReader.Delete()
    Success(data)
  }

  private[io] def readVTKAVSucdUnstructuredGrid(file: File): Try[vtkUnstructuredGrid] = {
    val vtkavsReader = new vtkAVSucdReader()
    vtkavsReader.SetFileName(file.getAbsolutePath)
    vtkavsReader.Update()
    val errCode = vtkavsReader.GetErrorCode()
    if (errCode != 0) {
      return Failure(new IOException(s"Could not read vtk UnstructuredGrid (received error code $errCode"))
    }
    val data = vtkavsReader.GetOutput()
    vtkavsReader.Delete()
    Success(data)
  }

  /**
   * Writes a [[TetrahedralMesh[_3D]]] to a file in one of the two standard VTK file formats ".vtk", or ".vtu".
   */
  def writeTetrahedralMesh(mesh: TetrahedralMesh[_3D], file: File): Try[Unit] = {
    val filename = file.getAbsolutePath
    val conversionFunction = (m: TetrahedralMesh[_3D]) =>
      TetrahedralMeshConversion.tetrahedralMeshToVTKUnstructuredGrid(m, None)
    filename match {
      case f if f.endsWith(".vtk") => writeToVTKFileThenDelete(mesh, writeVTKUgasVTK, conversionFunction, file)
      case f if f.endsWith(".vtu") => writeToVTKFileThenDelete(mesh, writeVTKUgasVTU, conversionFunction, file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  private[io] def writeToVTKFileThenDelete[T](volume: T,
                                              writeToFile: (vtkUnstructuredGrid, File) => Try[Unit],
                                              convertToVTKUG: T => vtkUnstructuredGrid,
                                              file: File
  ): Try[Unit] = {
    val vtkUg = convertToVTKUG(volume)
    for {
      result <- writeToFile(vtkUg, file)
    } yield {
      vtkUg.Delete()
      result
    }
  }

  private[io] def writeVTKUgasVTK(vtkUg: vtkUnstructuredGrid, file: File): Try[Unit] = {
    val writer = new vtkUnstructuredGridWriter()
    writer.SetFileName(file.getAbsolutePath)
    writer.SetInputData(vtkUg)
    writer.SetFileTypeToBinary()
    writer.Update()
    val succOrFailure = if (writer.GetErrorCode() != 0) {
      Failure(
        new IOException(s"could not write file ${file.getAbsolutePath} (received error code ${writer.GetErrorCode})")
      )
    } else {
      Success(())
    }
    writer.Delete()
    succOrFailure
  }

  private[io] def writeVTKUgasVTU(vtkUg: vtkUnstructuredGrid, file: File): Try[Unit] = {
    val writer = new vtkXMLUnstructuredGridWriter()
    writer.SetFileName(file.getAbsolutePath)
    writer.SetInputData(vtkUg)
    writer.SetDataModeToBinary()
    writer.Update()
    val succOrFailure = if (writer.GetErrorCode() != 0) {
      Failure(
        new IOException(s"could not write file ${file.getAbsolutePath} (received error code ${writer.GetErrorCode})")
      )
    } else {
      Success(())
    }
    writer.Delete()
    succOrFailure
  }

  def writeScalarVolumeMeshField[S: Scalar: ClassTag](meshData: ScalarVolumeMeshField[S], file: File): Try[Unit] = {
    val filename = file.getAbsolutePath
    val conversionFunction = (smf: ScalarVolumeMeshField[S]) =>
      TetrahedralMeshConversion.scalarVolumeMeshFieldToVtkUnstructuredGrid[S](smf, None)
    filename match {
      case f if f.endsWith(".vtk") => writeToVTKFileThenDelete(meshData, writeVTKUgasVTK, conversionFunction, file)
      case f if f.endsWith(".vtu") => writeToVTKFileThenDelete(meshData, writeVTKUgasVTU, conversionFunction, file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def writeMesh(mesh: TriangleMesh[_3D], file: File): Try[Unit] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") => writeVTK(mesh, file)
      case f if f.endsWith(".stl") => writeSTL(mesh, file)
      case f if f.endsWith(".ply") => writePLY(Left(mesh), file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  /**
   * Writes a [[VertexColorMesh3D]] to a file.
   *
   * **Important**: For PLY, since we use the VTK file writer, and since it does not support RGBA, only RGB, the alpha
   * channel will be ignored while writing.
   */
  def writeVertexColorMesh3D(mesh: VertexColorMesh3D, file: File): Try[Unit] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".ply") => writePLY(Right(mesh), file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def writeScalarMeshField[S: Scalar: ClassTag](meshData: ScalarMeshField[S], file: File): Try[Unit] = {
    val filename = file.getAbsolutePath
    filename match {
      case f if f.endsWith(".vtk") => writeVTK(meshData, file)
      case _ =>
        Failure(new IOException("Unknown file type received" + filename))
    }
  }

  def writeVTK[S: Scalar: ClassTag](meshData: ScalarMeshField[S], file: File): Try[Unit] = {
    val vtkPd = MeshConversion.scalarMeshFieldToVtkPolyData(meshData)
    val err = writeVTKPdasVTK(vtkPd, file)
    vtkPd.Delete()
    err
  }

  def writeVTK(surface: TriangleMesh[_3D], file: File): Try[Unit] = {
    val vtkPd = MeshConversion.meshToVtkPolyData(surface)
    val err = writeVTKPdasVTK(vtkPd, file)
    vtkPd.Delete()
    err
  }

  def writeSTL(surface: TriangleMesh[_3D], file: File): Try[Unit] = {
    val vtkPd = MeshConversion.meshToVtkPolyData(surface)
    val err = writeVTKPdAsSTL(vtkPd, file)
    vtkPd.Delete()
    err
  }

  private def writePLY(surface: Either[TriangleMesh[_3D], VertexColorMesh3D], file: File): Try[Unit] = {

    val vtkPd = surface match {
      case Right(colorMesh) => MeshConversion.meshToVtkPolyData(colorMesh.shape)
      case Left(shapeOnly)  => MeshConversion.meshToVtkPolyData(shapeOnly)
    }

    // add the colours if it is a vertex color
    surface match {
      case Right(colorMesh) => {

        val vtkColors = new vtkUnsignedCharArray()
        vtkColors.SetNumberOfComponents(4)

        // Add the three colors we have created to the array
        for (id <- colorMesh.shape.pointSet.pointIds) {
          val color = colorMesh.color(id)
          vtkColors.InsertNextTuple4((color.r * 255).toShort,
                                     (color.g * 255).toShort,
                                     (color.b * 255).toShort,
                                     color.a * 255
          )
        }
        vtkColors.SetName("RGBA")
        vtkPd.GetPointData().SetScalars(vtkColors)

      }
      case _ => {}
    }
    val writer = new vtkPLYWriter()
    writer.SetFileName(file.getAbsolutePath)
    writer.SetArrayName("RGBA")
    writer.SetComponent(0)
    writer.SetEnableAlpha(true)
    writer.SetInputData(vtkPd)
    writer.SetColorModeToDefault()
    writer.SetFileTypeToBinary()
    writer.Update()

    val succOrFailure = if (writer.GetErrorCode() != 0) {
      Failure(
        new IOException(s"could not write file ${file.getAbsolutePath} (received error code ${writer.GetErrorCode})")
      )
    } else {
      Success(())
    }
    writer.Delete()
    vtkPd.Delete()
    succOrFailure
  }

  private def writeVTKPdasVTK(vtkPd: vtkPolyData, file: File): Try[Unit] = {
    val writer = new vtkPolyDataWriter()
    writer.SetFileName(file.getAbsolutePath)
    writer.SetInputData(vtkPd)
    writer.SetFileTypeToBinary()
    writer.Update()
    val succOrFailure = if (writer.GetErrorCode() != 0) {
      Failure(
        new IOException(s"could not write file ${file.getAbsolutePath} (received error code ${writer.GetErrorCode})")
      )
    } else {
      Success(())
    }
    writer.Delete()
    succOrFailure
  }

  private def writeVTKPdAsSTL(vtkPd: vtkPolyData, file: File): Try[Unit] = {
    val writer = new vtkSTLWriter()
    writer.SetFileName(file.getAbsolutePath)
    writer.SetInputData(vtkPd)
    writer.SetFileTypeToBinary()
    writer.Update()
    val succOrFailure = if (writer.GetErrorCode() != 0) {
      Failure(
        new IOException(s"could not write file ${file.getAbsolutePath} (received error code ${writer.GetErrorCode})")
      )
    } else {
      Success(())
    }
    writer.Delete()
    succOrFailure
  }

  private def readVTKPolydata(file: File): Try[vtkPolyData] = {

    val vtkReader = new vtkPolyDataReader()
    vtkReader.SetFileName(file.getAbsolutePath)
    vtkReader.Update()
    val errCode = vtkReader.GetErrorCode()
    if (errCode != 0) {
      return Failure(new IOException(s"Could not read vtk mesh (received error code $errCode"))
    }
    val data = vtkReader.GetOutput()
    vtkReader.Delete()
    Success(data)
  }

  private def readVTK(file: File, correctMesh: Boolean = false): Try[TriangleMesh[_3D]] = {
    for {
      vtkPd <- readVTKPolydata(file)
      mesh <- {
        if (correctMesh) MeshConversion.vtkPolyDataToCorrectedTriangleMesh(vtkPd)
        else MeshConversion.vtkPolyDataToTriangleMesh(vtkPd)
      }
    } yield {
      vtkPd.Delete()
      mesh
    }
  }

  private def readSTL(file: File, correctMesh: Boolean = false): Try[TriangleMesh[_3D]] = {
    val stlReader = new vtkSTLReader()
    stlReader.SetFileName(file.getAbsolutePath)

    stlReader.MergingOn()

    // With the default point locator, it may happen that the stlReader merges
    // points that are very close by but not identical. To make sure that this never happens
    // we explicitly specify the tolerance.
    val pointLocator = new vtkMergePoints()
    pointLocator.SetTolerance(0.0)

    stlReader.SetLocator(pointLocator)
    stlReader.Update()
    val errCode = stlReader.GetErrorCode()
    if (errCode != 0) {
      return Failure(new IOException(s"Could not read stl mesh (received error code $errCode"))
    }

    val vtkPd = stlReader.GetOutput()
    val mesh =
      if (correctMesh) MeshConversion.vtkPolyDataToCorrectedTriangleMesh(vtkPd)
      else MeshConversion.vtkPolyDataToTriangleMesh(vtkPd)

    stlReader.Delete()
    vtkPd.Delete()
    mesh
  }

  private def getColorArray(polyData: vtkPolyData): Option[(String, vtkDataArray)] = {
    if (polyData.GetPointData() == null || polyData.GetPointData().GetNumberOfArrays() == 0) None
    else {
      val pointData = polyData.GetPointData()
      val pointDataArrays = for (i <- 0 until pointData.GetNumberOfArrays()) yield {
        (pointData.GetArrayName(i), pointData.GetArray(i))
      }
      pointDataArrays.find { case (name, array) => name == "RGB" || name == "RGBA" }
    }
  }

  private def readPLY(file: File): Try[Either[TriangleMesh[_3D], VertexColorMesh3D]] = {

    // read the ply header to find out if the ply is a textured mesh in ASCII (in which case we return a failure since VTKPLYReader Update() would crash otherwise)

    if (!file.exists()) {
      val filename = file.getCanonicalFile
      Failure(new IOException(s"Could not read ply file with name $filename. Reason: The file does not exist"))
    } else {
      val breader = new BufferedReader(new FileReader(file))
      val lineIterator = Iterator.continually(breader.readLine())

      val headerLines = lineIterator.dropWhile(_ != "ply").takeWhile(_ != "end_header").toIndexedSeq

      if (headerLines.exists(_.contains("TextureFile")) && headerLines.exists(_.contains("format ascii"))) {
        Failure(
          new IOException(
            "PLY file $filename seems to be a textured mesh in ASCII format which creates issues with the VTK ply reader. Please convert it to a binary ply or to a vertex color or shape only ply."
          )
        )
      } else {
        readPLYUsingVTK(file)
      }
    }
  }

  private def readPLYUsingVTK(file: File): Try[Either[TriangleMesh[_3D], VertexColorMesh3D]] = {
    val filename = file.getCanonicalFile
    val plyReader = new vtkPLYReader()
    plyReader.SetFileName(file.getAbsolutePath)
    plyReader.Update()

    val errCode = plyReader.GetErrorCode()
    if (errCode != 0) {
      return Failure(new IOException(s"Could not read ply mesh $filename (received VTK error code $errCode"))
    }

    val vtkPd = plyReader.GetOutput()
    val mesh = for {
      meshGeometry <- MeshConversion.vtkPolyDataToTriangleMesh(vtkPd)
    } yield {
      getColorArray(vtkPd) match {
        case Some(("RGBA", colorArray)) => {

          val colors = for (i <- 0 until colorArray.GetNumberOfTuples().toInt) yield {
            val rgba = colorArray.GetTuple4(i)
            RGBA(rgba(0) / 255.0, rgba(1) / 255.0, rgba(2) / 255.0, rgba(3) / 255.0)
          }
          Right(VertexColorMesh3D(meshGeometry, new SurfacePointProperty[RGBA](meshGeometry.triangulation, colors)))
        }
        case Some(("RGB", colorArray)) => {
          val colors = for (i <- 0 until colorArray.GetNumberOfTuples().toInt) yield {
            val rgb = colorArray.GetTuple3(i)
            RGBA(RGB(rgb(0) / 255.0, rgb(1) / 255.0, rgb(2) / 255.0))
          }
          Right(VertexColorMesh3D(meshGeometry, new SurfacePointProperty[RGBA](meshGeometry.triangulation, colors)))
        }
        case Some(_) => Left(meshGeometry)
        case None    => Left(meshGeometry)
      }
    }
    plyReader.Delete()
    vtkPd.Delete()
    mesh
  }

  private def readLineMeshVTK[D: NDSpace: LineMesh.Create: UnstructuredPoints.Create](file: File): Try[LineMesh[D]] = {
    val vtkReader = new vtkPolyDataReader()
    vtkReader.SetFileName(file.getAbsolutePath)
    vtkReader.Update()
    val errCode = vtkReader.GetErrorCode()
    if (errCode != 0) {
      return Failure(new IOException(s"Could not read vtk mesh (received error code $errCode"))
    }

    val vtkPd = vtkReader.GetOutput()
    val correctedMesh = for {
      polyline <- MeshConversion.vtkPolyDataToLineMesh[D](vtkPd)
    } yield {
      LineMesh.enforceConsistentCellDirections[D](polyline)
    }
    vtkReader.Delete()
    vtkPd.Delete()
    correctedMesh
  }

  private[this] def writeLineMeshVTK[D: NDSpace](mesh: LineMesh[D], file: File): Try[Unit] = {
    val vtkPd = MeshConversion.lineMeshToVTKPolyData(mesh)
    val err = writeVTKPdasVTK(vtkPd, file)
    vtkPd.Delete()
    err
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy