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