vtk9/IO/XML/vtkXMLPolyDataWriter.cxx

394 lines
13 KiB
C++

/*=========================================================================
Program: Visualization Toolkit
Module: vtkXMLPolyDataWriter.cxx
Copyright (c) Ken Martin, Will Schroeder, Bill Lorensen
All rights reserved.
See Copyright.txt or http://www.kitware.com/Copyright.htm for details.
This software is distributed WITHOUT ANY WARRANTY; without even
the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
PURPOSE. See the above copyright notice for more information.
=========================================================================*/
#include "vtkXMLPolyDataWriter.h"
#include "vtkCellArray.h"
#include "vtkCellData.h"
#include "vtkErrorCode.h"
#include "vtkIdTypeArray.h"
#include "vtkInformation.h"
#include "vtkInformationIntegerKey.h"
#include "vtkObjectFactory.h"
#include "vtkPointData.h"
#include "vtkPolyData.h"
#include "vtkStreamingDemandDrivenPipeline.h"
#define vtkXMLOffsetsManager_DoNotInclude
#include "vtkXMLOffsetsManager.h"
#undef vtkXMLOffsetsManager_DoNotInclude
vtkStandardNewMacro(vtkXMLPolyDataWriter);
//------------------------------------------------------------------------------
vtkXMLPolyDataWriter::vtkXMLPolyDataWriter()
{
this->VertsOM = new OffsetsManagerArray;
this->LinesOM = new OffsetsManagerArray;
this->StripsOM = new OffsetsManagerArray;
this->PolysOM = new OffsetsManagerArray;
}
//------------------------------------------------------------------------------
vtkXMLPolyDataWriter::~vtkXMLPolyDataWriter()
{
delete this->VertsOM;
delete this->LinesOM;
delete this->StripsOM;
delete this->PolysOM;
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::PrintSelf(ostream& os, vtkIndent indent)
{
this->Superclass::PrintSelf(os, indent);
}
//------------------------------------------------------------------------------
vtkPolyData* vtkXMLPolyDataWriter::GetInput()
{
return static_cast<vtkPolyData*>(this->Superclass::GetInput());
}
//------------------------------------------------------------------------------
const char* vtkXMLPolyDataWriter::GetDataSetName()
{
return "PolyData";
}
//------------------------------------------------------------------------------
const char* vtkXMLPolyDataWriter::GetDefaultFileExtension()
{
return "vtp";
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::AllocatePositionArrays()
{
this->Superclass::AllocatePositionArrays();
this->NumberOfVertsPositions = new unsigned long[this->NumberOfPieces];
this->NumberOfLinesPositions = new unsigned long[this->NumberOfPieces];
this->NumberOfStripsPositions = new unsigned long[this->NumberOfPieces];
this->NumberOfPolysPositions = new unsigned long[this->NumberOfPieces];
this->VertsOM->Allocate(this->NumberOfPieces, 2, this->NumberOfTimeSteps);
this->LinesOM->Allocate(this->NumberOfPieces, 2, this->NumberOfTimeSteps);
this->StripsOM->Allocate(this->NumberOfPieces, 2, this->NumberOfTimeSteps);
this->PolysOM->Allocate(this->NumberOfPieces, 2, this->NumberOfTimeSteps);
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::DeletePositionArrays()
{
this->Superclass::DeletePositionArrays();
delete[] this->NumberOfVertsPositions;
delete[] this->NumberOfLinesPositions;
delete[] this->NumberOfStripsPositions;
delete[] this->NumberOfPolysPositions;
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::WriteInlinePieceAttributes()
{
this->Superclass::WriteInlinePieceAttributes();
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
vtkPolyData* input = this->GetInput();
this->WriteScalarAttribute("NumberOfVerts", input->GetVerts()->GetNumberOfCells());
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->WriteScalarAttribute("NumberOfLines", input->GetLines()->GetNumberOfCells());
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->WriteScalarAttribute("NumberOfStrips", input->GetStrips()->GetNumberOfCells());
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->WriteScalarAttribute("NumberOfPolys", input->GetPolys()->GetNumberOfCells());
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::WriteInlinePiece(vtkIndent indent)
{
// Split progress range by the approximate fraction of data written
// by each step in this method.
float progressRange[2] = { 0.f, 0.f };
this->GetProgressRange(progressRange);
float fractions[6];
this->CalculateSuperclassFraction(fractions);
// Set the range of progress for superclass.
this->SetProgressRange(progressRange, 0, fractions);
// Let the superclass write its data.
this->Superclass::WriteInlinePiece(indent);
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
vtkPolyData* input = this->GetInput();
// Set the range of progress for Verts.
this->SetProgressRange(progressRange, 1, fractions);
// Write the Verts.
this->WriteCellsInline("Verts", input->GetVerts(), nullptr, indent);
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
// Set the range of progress for Lines.
this->SetProgressRange(progressRange, 2, fractions);
// Write the Lines.
this->WriteCellsInline("Lines", input->GetLines(), nullptr, indent);
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
// Set the range of progress for Strips.
this->SetProgressRange(progressRange, 3, fractions);
// Write the Strips.
this->WriteCellsInline("Strips", input->GetStrips(), nullptr, indent);
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
// Set the range of progress for Polys.
this->SetProgressRange(progressRange, 4, fractions);
// Write the Polys.
this->WriteCellsInline("Polys", input->GetPolys(), nullptr, indent);
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::WriteAppendedPieceAttributes(int index)
{
this->Superclass::WriteAppendedPieceAttributes(index);
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->NumberOfVertsPositions[index] = this->ReserveAttributeSpace("NumberOfVerts");
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->NumberOfLinesPositions[index] = this->ReserveAttributeSpace("NumberOfLines");
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->NumberOfStripsPositions[index] = this->ReserveAttributeSpace("NumberOfStrips");
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->NumberOfPolysPositions[index] = this->ReserveAttributeSpace("NumberOfPolys");
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::WriteAppendedPiece(int index, vtkIndent indent)
{
this->Superclass::WriteAppendedPiece(index, indent);
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->ConvertCells(this->GetInput()->GetVerts());
this->WriteCellsAppended("Verts", nullptr, indent, &this->VertsOM->GetPiece(index));
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->ConvertCells(this->GetInput()->GetLines());
this->WriteCellsAppended("Lines", nullptr, indent, &this->LinesOM->GetPiece(index));
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->ConvertCells(this->GetInput()->GetStrips());
this->WriteCellsAppended("Strips", nullptr, indent, &this->StripsOM->GetPiece(index));
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
this->ConvertCells(this->GetInput()->GetPolys());
this->WriteCellsAppended("Polys", nullptr, indent, &this->PolysOM->GetPiece(index));
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::WriteAppendedPieceData(int index)
{
ostream& os = *(this->Stream);
vtkPolyData* input = this->GetInput();
unsigned long returnPosition = os.tellp();
os.seekp(this->NumberOfVertsPositions[index]);
this->WriteScalarAttribute("NumberOfVerts", input->GetVerts()->GetNumberOfCells());
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
os.seekp(this->NumberOfLinesPositions[index]);
this->WriteScalarAttribute("NumberOfLines", input->GetLines()->GetNumberOfCells());
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
os.seekp(this->NumberOfStripsPositions[index]);
this->WriteScalarAttribute("NumberOfStrips", input->GetStrips()->GetNumberOfCells());
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
os.seekp(this->NumberOfPolysPositions[index]);
this->WriteScalarAttribute("NumberOfPolys", input->GetPolys()->GetNumberOfCells());
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
os.seekp(returnPosition);
// Split progress range by the approximate fraction of data written
// by each step in this method.
float progressRange[2] = { 0.f, 0.f };
this->GetProgressRange(progressRange);
float fractions[6];
this->CalculateSuperclassFraction(fractions);
// Set the range of progress for superclass.
this->SetProgressRange(progressRange, 0, fractions);
// Let the superclass write its data.
this->Superclass::WriteAppendedPieceData(index);
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
// Set the range of progress for Verts.
this->SetProgressRange(progressRange, 1, fractions);
// Write the Verts.
this->WriteCellsAppendedData(
input->GetVerts(), nullptr, this->CurrentTimeIndex, &this->VertsOM->GetPiece(index));
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
// Set the range of progress for Lines.
this->SetProgressRange(progressRange, 2, fractions);
// Write the Lines.
this->WriteCellsAppendedData(
input->GetLines(), nullptr, this->CurrentTimeIndex, &this->LinesOM->GetPiece(index));
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
// Set the range of progress for Strips.
this->SetProgressRange(progressRange, 3, fractions);
// Write the Strips.
this->WriteCellsAppendedData(
input->GetStrips(), nullptr, this->CurrentTimeIndex, &this->StripsOM->GetPiece(index));
if (this->ErrorCode == vtkErrorCode::OutOfDiskSpaceError)
{
return;
}
// Set the range of progress for Polys.
this->SetProgressRange(progressRange, 4, fractions);
// Write the Polys.
this->WriteCellsAppendedData(
input->GetPolys(), nullptr, this->CurrentTimeIndex, &this->PolysOM->GetPiece(index));
}
//------------------------------------------------------------------------------
vtkIdType vtkXMLPolyDataWriter::GetNumberOfInputCells()
{
vtkPolyData* input = this->GetInput();
return (input->GetVerts()->GetNumberOfCells() + input->GetLines()->GetNumberOfCells() +
input->GetStrips()->GetNumberOfCells() + input->GetPolys()->GetNumberOfCells());
}
//------------------------------------------------------------------------------
void vtkXMLPolyDataWriter::CalculateSuperclassFraction(float* fractions)
{
vtkPolyData* input = this->GetInput();
// The superclass will write point/cell data and point specifications.
int pdArrays = input->GetPointData()->GetNumberOfArrays();
int cdArrays = input->GetCellData()->GetNumberOfArrays();
vtkIdType pdSize = pdArrays * this->GetNumberOfInputPoints();
vtkIdType cdSize = cdArrays * this->GetNumberOfInputCells();
vtkIdType pointsSize = this->GetNumberOfInputPoints();
// This class will write cell specifications.
vtkIdType connectSizeV = input->GetVerts()->GetNumberOfConnectivityIds();
vtkIdType connectSizeL = input->GetLines()->GetNumberOfConnectivityIds();
vtkIdType connectSizeS = input->GetStrips()->GetNumberOfConnectivityIds();
vtkIdType connectSizeP = input->GetPolys()->GetNumberOfConnectivityIds();
vtkIdType offsetSizeV = input->GetVerts()->GetNumberOfCells();
vtkIdType offsetSizeL = input->GetLines()->GetNumberOfCells();
vtkIdType offsetSizeS = input->GetStrips()->GetNumberOfCells();
vtkIdType offsetSizeP = input->GetPolys()->GetNumberOfCells();
fractions[0] = 0;
fractions[1] = fractions[0] + pdSize + cdSize + pointsSize;
fractions[2] = fractions[1] + connectSizeV + offsetSizeV;
fractions[3] = fractions[2] + connectSizeL + offsetSizeL;
fractions[4] = fractions[3] + connectSizeS + offsetSizeS;
fractions[5] = fractions[4] + connectSizeP + offsetSizeP;
if (fractions[5] == 0)
{
fractions[5] = 1;
}
for (int i = 0; i < 5; ++i)
{
fractions[i + 1] = fractions[i + 1] / fractions[5];
}
}
//------------------------------------------------------------------------------
int vtkXMLPolyDataWriter::FillInputPortInformation(int, vtkInformation* info)
{
info->Set(vtkAlgorithm::INPUT_REQUIRED_DATA_TYPE(), "vtkPolyData");
return 1;
}