You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
241 lines
7.2 KiB
241 lines
7.2 KiB
/*=========================================================================
|
|
|
|
Program: Visualization Toolkit
|
|
Module: $RCSfile: vtkEdgePoints.cxx,v $
|
|
|
|
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 "vtkEdgePoints.h"
|
|
|
|
#include "vtkCell.h"
|
|
#include "vtkCellArray.h"
|
|
#include "vtkCellData.h"
|
|
#include "vtkDataSet.h"
|
|
#include "vtkGenericCell.h"
|
|
#include "vtkInformation.h"
|
|
#include "vtkInformationVector.h"
|
|
#include "vtkMergePoints.h"
|
|
#include "vtkObjectFactory.h"
|
|
#include "vtkPointData.h"
|
|
#include "vtkPolyData.h"
|
|
|
|
vtkCxxRevisionMacro(vtkEdgePoints, "$Revision: 1.57 $");
|
|
vtkStandardNewMacro(vtkEdgePoints);
|
|
|
|
// Construct object with contour value of 0.0.
|
|
vtkEdgePoints::vtkEdgePoints()
|
|
{
|
|
this->Value = 0.0;
|
|
this->Locator = vtkMergePoints::New();
|
|
}
|
|
|
|
vtkEdgePoints::~vtkEdgePoints()
|
|
{
|
|
this->Locator->Delete();
|
|
this->Locator = NULL;
|
|
}
|
|
|
|
//
|
|
// General filter: handles arbitrary input.
|
|
//
|
|
int vtkEdgePoints::RequestData(
|
|
vtkInformation *vtkNotUsed(request),
|
|
vtkInformationVector **inputVector,
|
|
vtkInformationVector *outputVector)
|
|
{
|
|
// get the info objects
|
|
vtkInformation *inInfo = inputVector[0]->GetInformationObject(0);
|
|
vtkInformation *outInfo = outputVector->GetInformationObject(0);
|
|
|
|
// get the input and ouptut
|
|
vtkDataSet *input = vtkDataSet::SafeDownCast(
|
|
inInfo->Get(vtkDataObject::DATA_OBJECT()));
|
|
vtkPolyData *output = vtkPolyData::SafeDownCast(
|
|
outInfo->Get(vtkDataObject::DATA_OBJECT()));
|
|
|
|
vtkDataArray *inScalars;
|
|
vtkPoints *newPts;
|
|
vtkCellArray *newVerts;
|
|
vtkIdType cellId, ptId, edgeId, newCellId;
|
|
int above, below, i, numEdges;
|
|
vtkCell *edge;
|
|
double range[2];
|
|
double s0, s1, x0[3], x1[3], x[3], t;
|
|
double e0Scalar, deltaScalar;
|
|
int e0, e1;
|
|
vtkIdType p1, p2;
|
|
vtkIdType pts[1];
|
|
vtkIdType numCells, estimatedSize;
|
|
vtkDataArray *cellScalars;
|
|
vtkPointData *inPd=input->GetPointData(), *outPd=output->GetPointData();
|
|
vtkCellData *inCd=input->GetCellData(), *outCd=output->GetCellData();
|
|
|
|
vtkDebugMacro(<< "Generating edge points");
|
|
|
|
// Initialize and check input
|
|
//
|
|
if ( ! (inScalars = input->GetPointData()->GetScalars()) )
|
|
{
|
|
vtkErrorMacro(<<"No scalar data to contour");
|
|
return 1;
|
|
}
|
|
|
|
inScalars->GetRange(range,0);
|
|
if ( this->Value < range[0] || this->Value > range[1] )
|
|
{
|
|
vtkWarningMacro(<<"Value lies outside of scalar range");
|
|
return 1;
|
|
}
|
|
|
|
numCells = input->GetNumberOfCells();
|
|
estimatedSize = (vtkIdType) (numCells * .75);
|
|
estimatedSize = estimatedSize / 1024 * 1024; //multiple of 1024
|
|
if (estimatedSize < 1024)
|
|
{
|
|
estimatedSize = 1024;
|
|
}
|
|
|
|
newPts = vtkPoints::New();
|
|
newPts->Allocate(estimatedSize, estimatedSize/2);
|
|
newVerts = vtkCellArray::New();
|
|
newVerts->Allocate(estimatedSize, estimatedSize/2);
|
|
cellScalars = inScalars->NewInstance();
|
|
cellScalars->SetNumberOfComponents(inScalars->GetNumberOfComponents());
|
|
cellScalars->Allocate(VTK_CELL_SIZE*inScalars->GetNumberOfComponents());
|
|
|
|
this->Locator->InitPointInsertion (newPts, input->GetBounds());
|
|
|
|
// interpolate data along edge; copy cell data
|
|
outPd->InterpolateAllocate(inPd,5000,10000);
|
|
outCd->CopyAllocate(inCd,5000,10000);
|
|
|
|
// Traverse all edges. Since edges are not explicitly represented, use a
|
|
// trick: traverse all cells and obtain cell edges and then cell edge
|
|
// neighbors. If cell id < all edge neigbors ids, then this edge has not
|
|
// yet been visited and is processed.
|
|
//
|
|
int abort=0;
|
|
vtkIdType progressInterval = numCells/20 + 1;
|
|
vtkGenericCell *cell = vtkGenericCell::New();
|
|
for (cellId=0; cellId < numCells && !abort; cellId++)
|
|
{
|
|
if ( ! (cellId % progressInterval) )
|
|
{
|
|
vtkDebugMacro(<<"Processing #" << cellId);
|
|
this->UpdateProgress ((double)cellId/numCells);
|
|
abort = this->GetAbortExecute();
|
|
}
|
|
|
|
input->GetCell(cellId,cell);
|
|
inScalars->GetTuples(cell->PointIds, cellScalars);
|
|
|
|
// loop over cell points to check if cell straddles isosurface value
|
|
for ( above=below=0, ptId=0; ptId < cell->GetNumberOfPoints(); ptId++ )
|
|
{
|
|
if ( cellScalars->GetComponent(ptId,0) >= this->Value )
|
|
{
|
|
above = 1;
|
|
}
|
|
else if ( cellScalars->GetComponent(ptId,0) < this->Value )
|
|
{
|
|
below = 1;
|
|
}
|
|
}
|
|
|
|
if ( above && below ) //contour passes through cell
|
|
{
|
|
if ( cell->GetCellDimension() < 2 ) //only points can be generated
|
|
{
|
|
cell->Contour(this->Value, cellScalars, this->Locator, newVerts,
|
|
NULL, NULL, inPd, outPd, inCd, cellId, outCd);
|
|
}
|
|
|
|
else //
|
|
{
|
|
numEdges = cell->GetNumberOfEdges();
|
|
for (edgeId=0; edgeId < numEdges; edgeId++)
|
|
{
|
|
edge = cell->GetEdge(edgeId);
|
|
inScalars->GetTuples(edge->PointIds, cellScalars);
|
|
|
|
s0 = cellScalars->GetComponent(0,0);
|
|
s1 = cellScalars->GetComponent(1,0);
|
|
if ( (s0 < this->Value && s1 >= this->Value) ||
|
|
(s0 >= this->Value && s1 < this->Value) )
|
|
{
|
|
//ordering intersection direction avoids numerical problems
|
|
deltaScalar = s1 - s0;
|
|
if (deltaScalar > 0)
|
|
{
|
|
e0 = 0; e1 = 1;
|
|
e0Scalar = s0;
|
|
}
|
|
else
|
|
{
|
|
e0 = 1; e1 = 0;
|
|
e0Scalar = s1;
|
|
deltaScalar = -deltaScalar;
|
|
}
|
|
|
|
t = (this->Value - e0Scalar) / deltaScalar;
|
|
|
|
edge->Points->GetPoint(e0,x0);
|
|
edge->Points->GetPoint(e1,x1);
|
|
|
|
for (i=0; i<3; i++)
|
|
{
|
|
x[i] = x0[i] + t * (x1[i] - x0[i]);
|
|
}
|
|
if ( this->Locator->InsertUniquePoint(x, pts[0]) )
|
|
{
|
|
newCellId = newVerts->InsertNextCell(1,pts);
|
|
outCd->CopyData(inCd,cellId,newCellId);
|
|
p1 = edge->PointIds->GetId(e0);
|
|
p2 = edge->PointIds->GetId(e1);
|
|
outPd->InterpolateEdge(inPd,pts[0],p1,p2,t);
|
|
} //if point not created before
|
|
} //if edge straddles contour value
|
|
} //for each edge
|
|
} //dimension 2 and higher
|
|
} //above and below
|
|
} //for all cells
|
|
cell->Delete();
|
|
|
|
vtkDebugMacro(<<"Created: " << newPts->GetNumberOfPoints() << " points");
|
|
|
|
// Update ourselves. Because we don't know up front how many verts we've
|
|
// created, take care to reclaim memory.
|
|
//
|
|
output->SetPoints(newPts);
|
|
newPts->Delete();
|
|
|
|
output->SetVerts(newVerts);
|
|
newVerts->Delete();
|
|
|
|
this->Locator->Initialize();//free up any extra memory
|
|
output->Squeeze();
|
|
|
|
cellScalars->Delete();
|
|
|
|
return 1;
|
|
}
|
|
|
|
int vtkEdgePoints::FillInputPortInformation(int, vtkInformation *info)
|
|
{
|
|
info->Set(vtkAlgorithm::INPUT_REQUIRED_DATA_TYPE(), "vtkDataSet");
|
|
return 1;
|
|
}
|
|
|
|
void vtkEdgePoints::PrintSelf(ostream& os, vtkIndent indent)
|
|
{
|
|
this->Superclass::PrintSelf(os,indent);
|
|
|
|
os << indent << "Contour Value: " << this->Value << "\n";
|
|
}
|
|
|