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.
585 lines
16 KiB
585 lines
16 KiB
/*=========================================================================
|
|
|
|
Program: Visualization Toolkit
|
|
Module: $RCSfile: vtkGaussianSplatter.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 "vtkGaussianSplatter.h"
|
|
|
|
#include "vtkDoubleArray.h"
|
|
#include "vtkImageData.h"
|
|
#include "vtkInformation.h"
|
|
#include "vtkInformationVector.h"
|
|
#include "vtkObjectFactory.h"
|
|
#include "vtkStreamingDemandDrivenPipeline.h"
|
|
#include "vtkPointData.h"
|
|
|
|
#include <math.h>
|
|
|
|
vtkCxxRevisionMacro(vtkGaussianSplatter, "$Revision: 1.59 $");
|
|
vtkStandardNewMacro(vtkGaussianSplatter);
|
|
|
|
// Construct object with dimensions=(50,50,50); automatic computation of
|
|
// bounds; a splat radius of 0.1; an exponent factor of -5; and normal and
|
|
// scalar warping turned on.
|
|
vtkGaussianSplatter::vtkGaussianSplatter()
|
|
{
|
|
this->SampleDimensions[0] = 50;
|
|
this->SampleDimensions[1] = 50;
|
|
this->SampleDimensions[2] = 50;
|
|
|
|
this->Radius = 0.1;
|
|
this->ExponentFactor = -5.0;
|
|
|
|
this->ModelBounds[0] = 0.0;
|
|
this->ModelBounds[1] = 0.0;
|
|
this->ModelBounds[2] = 0.0;
|
|
this->ModelBounds[3] = 0.0;
|
|
this->ModelBounds[4] = 0.0;
|
|
this->ModelBounds[5] = 0.0;
|
|
|
|
this->NormalWarping = 1;
|
|
this->Eccentricity = 2.5;
|
|
|
|
this->ScalarWarping = 1;
|
|
this->ScaleFactor = 1.0;
|
|
|
|
this->Capping = 1;
|
|
this->CapValue = 0.0;
|
|
|
|
this->AccumulationMode = VTK_ACCUMULATION_MODE_MAX;
|
|
this->NullValue = 0.0;
|
|
}
|
|
|
|
int vtkGaussianSplatter::RequestInformation (
|
|
vtkInformation * vtkNotUsed(request),
|
|
vtkInformationVector ** vtkNotUsed( inputVector ),
|
|
vtkInformationVector *outputVector)
|
|
{
|
|
// get the info objects
|
|
vtkInformation* outInfo = outputVector->GetInformationObject(0);
|
|
|
|
// use model bounds if set
|
|
this->Origin[0] = 0;
|
|
this->Origin[1] = 0;
|
|
this->Origin[2] = 0;
|
|
if ( this->ModelBounds[0] < this->ModelBounds[1] &&
|
|
this->ModelBounds[2] < this->ModelBounds[3] &&
|
|
this->ModelBounds[4] < this->ModelBounds[5] )
|
|
{
|
|
this->Origin[0] = this->ModelBounds[0];
|
|
this->Origin[1] = this->ModelBounds[2];
|
|
this->Origin[2] = this->ModelBounds[4];
|
|
}
|
|
|
|
outInfo->Set(vtkDataObject::ORIGIN(), this->Origin, 3);
|
|
|
|
int i;
|
|
for (i=0; i<3; i++)
|
|
{
|
|
this->Spacing[i] = (this->ModelBounds[2*i+1] - this->ModelBounds[2*i])
|
|
/ (this->SampleDimensions[i] - 1);
|
|
if ( this->Spacing[i] <= 0.0 )
|
|
{
|
|
this->Spacing[i] = 1.0;
|
|
}
|
|
}
|
|
outInfo->Set(vtkDataObject::SPACING(),this->Spacing,3);
|
|
|
|
outInfo->Set(vtkStreamingDemandDrivenPipeline::WHOLE_EXTENT(),
|
|
0, this->SampleDimensions[0] - 1,
|
|
0, this->SampleDimensions[1] - 1,
|
|
0, this->SampleDimensions[2] - 1);
|
|
vtkDataObject::SetPointDataActiveScalarInfo(outInfo, VTK_DOUBLE, 1);
|
|
return 1;
|
|
}
|
|
|
|
int vtkGaussianSplatter::RequestData(
|
|
vtkInformation* vtkNotUsed( request ),
|
|
vtkInformationVector** inputVector,
|
|
vtkInformationVector* outputVector)
|
|
{
|
|
// get the data object
|
|
vtkInformation *outInfo = outputVector->GetInformationObject(0);
|
|
vtkImageData *output = vtkImageData::SafeDownCast(
|
|
outInfo->Get(vtkDataObject::DATA_OBJECT()));
|
|
|
|
output->SetExtent(
|
|
outInfo->Get(vtkStreamingDemandDrivenPipeline::WHOLE_EXTENT()));
|
|
output->AllocateScalars();
|
|
|
|
vtkIdType numPts, numNewPts, ptId, idx, i;
|
|
int j, k;
|
|
int min[3], max[3];
|
|
vtkPointData *pd;
|
|
vtkDataArray *inNormals=NULL;
|
|
vtkDataArray *inScalars=NULL;
|
|
double loc[3], dist2, cx[3];
|
|
vtkDoubleArray *newScalars =
|
|
vtkDoubleArray::SafeDownCast(output->GetPointData()->GetScalars());
|
|
|
|
vtkInformation *inInfo = inputVector[0]->GetInformationObject(0);
|
|
vtkDataSet *input = vtkDataSet::SafeDownCast(
|
|
inInfo->Get(vtkDataObject::DATA_OBJECT()));
|
|
int sliceSize=this->SampleDimensions[0]*this->SampleDimensions[1];
|
|
|
|
vtkDebugMacro(<< "Splatting data");
|
|
|
|
// Make sure points are available
|
|
//
|
|
if ( (numPts=input->GetNumberOfPoints()) < 1 )
|
|
{
|
|
vtkErrorMacro(<<"No points to splat!");
|
|
return 1;
|
|
}
|
|
|
|
// Compute the radius of influence of the points. If an
|
|
// automatically generated bounding box has been generated, increase
|
|
// its size slightly to acoomodate the radius of influence.
|
|
//
|
|
this->Eccentricity2 = this->Eccentricity * this->Eccentricity;
|
|
|
|
numNewPts = this->SampleDimensions[0] * this->SampleDimensions[1] *
|
|
this->SampleDimensions[2];
|
|
for (i=0; i<numNewPts; i++)
|
|
{
|
|
newScalars->SetTuple(i,&this->NullValue);
|
|
}
|
|
this->Visited = new char[numNewPts];
|
|
for (i=0; i < numNewPts; i++)
|
|
{
|
|
this->Visited[i] = 0;
|
|
}
|
|
|
|
output->SetDimensions(this->GetSampleDimensions());
|
|
this->ComputeModelBounds(input,output, outInfo);
|
|
|
|
// Set up function pointers to sample functions
|
|
//
|
|
pd = input->GetPointData();
|
|
if ( this->NormalWarping && (inNormals=pd->GetNormals()) != NULL )
|
|
{
|
|
this->Sample = &vtkGaussianSplatter::EccentricGaussian;
|
|
}
|
|
else
|
|
{
|
|
this->Sample = &vtkGaussianSplatter::Gaussian;
|
|
}
|
|
|
|
if ( this->ScalarWarping && (inScalars=pd->GetScalars()) != NULL )
|
|
{
|
|
this->SampleFactor = &vtkGaussianSplatter::ScalarSampling;
|
|
}
|
|
else
|
|
{
|
|
this->SampleFactor = &vtkGaussianSplatter::PositionSampling;
|
|
this->S = 0.0; //position sampling does not require S to be defined
|
|
//but this makes purify happy.
|
|
}
|
|
|
|
// Traverse all points - splatting each into the volume.
|
|
// For each point, determine which voxel it is in. Then determine
|
|
// the subvolume that the splat is contained in, and process that.
|
|
//
|
|
int abortExecute=0;
|
|
vtkIdType progressInterval = numPts/20 + 1;
|
|
for (ptId=0; ptId < numPts && !abortExecute; ptId++)
|
|
{
|
|
if ( ! (ptId % progressInterval) )
|
|
{
|
|
vtkDebugMacro(<<"Inserting point #" << ptId);
|
|
this->UpdateProgress ((double)ptId/numPts);
|
|
abortExecute = this->GetAbortExecute();
|
|
}
|
|
|
|
this->P = input->GetPoint(ptId);
|
|
if ( inNormals != NULL )
|
|
{
|
|
this->N = inNormals->GetTuple(ptId);
|
|
}
|
|
if ( inScalars != NULL )
|
|
{
|
|
this->S = inScalars->GetComponent(ptId,0);
|
|
}
|
|
|
|
// Determine the voxel that the point is in
|
|
for (i=0; i<3; i++)
|
|
{
|
|
loc[i] = (this->P[i] - this->Origin[i]) / this->Spacing[i];
|
|
}
|
|
|
|
// Determine splat footprint
|
|
for (i=0; i<3; i++)
|
|
{
|
|
min[i] = (int) floor((double)loc[i]-this->SplatDistance[i]);
|
|
max[i] = (int) ceil((double)loc[i]+this->SplatDistance[i]);
|
|
if ( min[i] < 0 )
|
|
{
|
|
min[i] = 0;
|
|
}
|
|
if ( max[i] >= this->SampleDimensions[i] )
|
|
{
|
|
max[i] = this->SampleDimensions[i] - 1;
|
|
}
|
|
}
|
|
|
|
// Loop over all sample points in volume within footprint and
|
|
// evaluate the splat
|
|
for (k=min[2]; k<=max[2]; k++)
|
|
{
|
|
cx[2] = this->Origin[2] + this->Spacing[2]*k;
|
|
for (j=min[1]; j<=max[1]; j++)
|
|
{
|
|
cx[1] = this->Origin[1] + this->Spacing[1]*j;
|
|
for (i=min[0]; i<=max[0]; i++)
|
|
{
|
|
cx[0] = this->Origin[0] + this->Spacing[0]*i;
|
|
if ( (dist2=(this->*Sample)(cx)) <= this->Radius2 )
|
|
{
|
|
idx = i + j*this->SampleDimensions[0] + k*sliceSize;
|
|
this->SetScalar(idx,dist2, newScalars);
|
|
}//if within splat radius
|
|
}
|
|
}
|
|
}//within splat footprint
|
|
}//for all input points
|
|
|
|
// If capping is turned on, set the distances of the outside of the volume
|
|
// to the CapValue.
|
|
//
|
|
if ( this->Capping )
|
|
{
|
|
this->Cap(newScalars);
|
|
}
|
|
|
|
vtkDebugMacro(<< "Splatted " << input->GetNumberOfPoints() << " points");
|
|
|
|
// Update self and release memeory
|
|
//
|
|
delete [] this->Visited;
|
|
|
|
return 1;
|
|
}
|
|
|
|
// Compute the size of the sample bounding box automatically from the
|
|
// input data.
|
|
void vtkGaussianSplatter::ComputeModelBounds(vtkDataSet *input,
|
|
vtkImageData *output,
|
|
vtkInformation *outInfo)
|
|
{
|
|
double *bounds, maxDist;
|
|
int i, adjustBounds=0;
|
|
|
|
// compute model bounds if not set previously
|
|
if ( this->ModelBounds[0] >= this->ModelBounds[1] ||
|
|
this->ModelBounds[2] >= this->ModelBounds[3] ||
|
|
this->ModelBounds[4] >= this->ModelBounds[5] )
|
|
{
|
|
adjustBounds = 1;
|
|
bounds = input->GetBounds();
|
|
}
|
|
else
|
|
{
|
|
bounds = this->ModelBounds;
|
|
}
|
|
|
|
for (maxDist=0.0, i=0; i<3; i++)
|
|
{
|
|
if ( (bounds[2*i+1] - bounds[2*i]) > maxDist )
|
|
{
|
|
maxDist = bounds[2*i+1] - bounds[2*i];
|
|
}
|
|
}
|
|
maxDist *= this->Radius;
|
|
this->Radius2 = maxDist * maxDist;
|
|
|
|
// adjust bounds so model fits strictly inside (only if not set previously)
|
|
if ( adjustBounds )
|
|
{
|
|
for (i=0; i<3; i++)
|
|
{
|
|
this->ModelBounds[2*i] = bounds[2*i] - maxDist;
|
|
this->ModelBounds[2*i+1] = bounds[2*i+1] + maxDist;
|
|
}
|
|
}
|
|
|
|
// Set volume origin and data spacing
|
|
outInfo->Set(vtkDataObject::ORIGIN(),
|
|
this->ModelBounds[0],this->ModelBounds[2],
|
|
this->ModelBounds[4]);
|
|
memcpy(this->Origin,outInfo->Get(vtkDataObject::ORIGIN()), sizeof(double)*6);
|
|
output->SetOrigin(this->Origin);
|
|
|
|
for (i=0; i<3; i++)
|
|
{
|
|
this->Spacing[i] = (this->ModelBounds[2*i+1] - this->ModelBounds[2*i])
|
|
/ (this->SampleDimensions[i] - 1);
|
|
if ( this->Spacing[i] <= 0.0 )
|
|
{
|
|
this->Spacing[i] = 1.0;
|
|
}
|
|
}
|
|
outInfo->Set(vtkDataObject::SPACING(),this->Spacing,3);
|
|
output->SetSpacing(this->Spacing);
|
|
|
|
// Determine the splat propagation distance...used later
|
|
for (i=0; i<3; i++)
|
|
{
|
|
this->SplatDistance[i] = maxDist / this->Spacing[i];
|
|
}
|
|
}
|
|
|
|
// Set the dimensions of the sampling structured point set.
|
|
void vtkGaussianSplatter::SetSampleDimensions(int i, int j, int k)
|
|
{
|
|
int dim[3];
|
|
|
|
dim[0] = i;
|
|
dim[1] = j;
|
|
dim[2] = k;
|
|
|
|
this->SetSampleDimensions(dim);
|
|
}
|
|
|
|
void vtkGaussianSplatter::SetSampleDimensions(int dim[3])
|
|
{
|
|
int dataDim, i;
|
|
|
|
vtkDebugMacro(<< " setting SampleDimensions to (" << dim[0] << ","
|
|
<< dim[1] << "," << dim[2] << ")");
|
|
|
|
if (dim[0] != this->SampleDimensions[0] ||
|
|
dim[1] != this->SampleDimensions[1] ||
|
|
dim[2] != this->SampleDimensions[2] )
|
|
{
|
|
if ( dim[0]<1 || dim[1]<1 || dim[2]<1 )
|
|
{
|
|
vtkErrorMacro (<< "Bad Sample Dimensions, retaining previous values");
|
|
return;
|
|
}
|
|
|
|
for (dataDim=0, i=0; i<3 ; i++)
|
|
{
|
|
if (dim[i] > 1)
|
|
{
|
|
dataDim++;
|
|
}
|
|
}
|
|
|
|
if ( dataDim < 3 )
|
|
{
|
|
vtkErrorMacro(<<"Sample dimensions must define a volume!");
|
|
return;
|
|
}
|
|
|
|
for ( i=0; i<3; i++)
|
|
{
|
|
this->SampleDimensions[i] = dim[i];
|
|
}
|
|
|
|
this->Modified();
|
|
}
|
|
}
|
|
|
|
void vtkGaussianSplatter::Cap(vtkDoubleArray *s)
|
|
{
|
|
int i,j,k;
|
|
vtkIdType idx;
|
|
int d01=this->SampleDimensions[0]*this->SampleDimensions[1];
|
|
|
|
// i-j planes
|
|
//k = 0;
|
|
for (j=0; j<this->SampleDimensions[1]; j++)
|
|
{
|
|
for (i=0; i<this->SampleDimensions[0]; i++)
|
|
{
|
|
s->SetTuple(i+j*this->SampleDimensions[0], &this->CapValue);
|
|
}
|
|
}
|
|
k = this->SampleDimensions[2] - 1;
|
|
idx = k*d01;
|
|
for (j=0; j<this->SampleDimensions[1]; j++)
|
|
{
|
|
for (i=0; i<this->SampleDimensions[0]; i++)
|
|
{
|
|
s->SetTuple(idx+i+j*this->SampleDimensions[0], &this->CapValue);
|
|
}
|
|
}
|
|
// j-k planes
|
|
//i = 0;
|
|
for (k=0; k<this->SampleDimensions[2]; k++)
|
|
{
|
|
for (j=0; j<this->SampleDimensions[1]; j++)
|
|
{
|
|
s->SetTuple(j*this->SampleDimensions[0]+k*d01, &this->CapValue);
|
|
}
|
|
}
|
|
i = this->SampleDimensions[0] - 1;
|
|
for (k=0; k<this->SampleDimensions[2]; k++)
|
|
{
|
|
for (j=0; j<this->SampleDimensions[1]; j++)
|
|
{
|
|
s->SetTuple(i+j*this->SampleDimensions[0]+k*d01, &this->CapValue);
|
|
}
|
|
}
|
|
// i-k planes
|
|
//j = 0;
|
|
for (k=0; k<this->SampleDimensions[2]; k++)
|
|
{
|
|
for (i=0; i<this->SampleDimensions[0]; i++)
|
|
{
|
|
s->SetTuple(i+k*d01, &this->CapValue);
|
|
}
|
|
}
|
|
j = this->SampleDimensions[1] - 1;
|
|
idx = j*this->SampleDimensions[0];
|
|
for (k=0; k<this->SampleDimensions[2]; k++)
|
|
{
|
|
for (i=0; i<this->SampleDimensions[0]; i++)
|
|
{
|
|
s->SetTuple(idx+i+k*d01, &this->CapValue);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Gaussian sampling
|
|
//
|
|
double vtkGaussianSplatter::Gaussian (double cx[3])
|
|
{
|
|
return ((cx[0]-P[0])*(cx[0]-P[0]) + (cx[1]-P[1])*(cx[1]-P[1]) +
|
|
(cx[2]-P[2])*(cx[2]-P[2]) );
|
|
}
|
|
|
|
//
|
|
// Ellipsoidal Gaussian sampling
|
|
//
|
|
double vtkGaussianSplatter::EccentricGaussian (double cx[3])
|
|
{
|
|
double v[3], r2, z2, rxy2, mag;
|
|
|
|
v[0] = cx[0] - this->P[0];
|
|
v[1] = cx[1] - this->P[1];
|
|
v[2] = cx[2] - this->P[2];
|
|
|
|
r2 = v[0]*v[0] + v[1]*v[1] + v[2]*v[2];
|
|
|
|
if ( (mag=this->N[0]*this->N[0]+
|
|
this->N[1]*this->N[1]+
|
|
this->N[2]*this->N[2]) != 1.0 )
|
|
{
|
|
if ( mag == 0.0 )
|
|
{
|
|
mag = 1.0;
|
|
}
|
|
else
|
|
{
|
|
mag = sqrt((double)mag);
|
|
}
|
|
}
|
|
|
|
z2 = (v[0]*this->N[0] + v[1]*this->N[1] + v[2]*this->N[2])/mag;
|
|
z2 = z2*z2;
|
|
|
|
rxy2 = r2 - z2;
|
|
|
|
return (rxy2/this->Eccentricity2 + z2);
|
|
}
|
|
|
|
void vtkGaussianSplatter::SetScalar(int idx, double dist2,
|
|
vtkDoubleArray *newScalars)
|
|
{
|
|
double v = (this->*SampleFactor)(this->S) * exp((double)
|
|
(this->ExponentFactor*(dist2)/(this->Radius2)));
|
|
|
|
if ( ! this->Visited[idx] )
|
|
{
|
|
this->Visited[idx] = 1;
|
|
newScalars->SetTuple(idx,&v);
|
|
}
|
|
else
|
|
{
|
|
double s = newScalars->GetValue(idx);
|
|
switch (this->AccumulationMode)
|
|
{
|
|
case VTK_ACCUMULATION_MODE_MIN:
|
|
newScalars->SetTuple(idx,(s < v ? &s : &v));
|
|
break;
|
|
case VTK_ACCUMULATION_MODE_MAX:
|
|
newScalars->SetTuple(idx,(s > v ? &s : &v));
|
|
break;
|
|
case VTK_ACCUMULATION_MODE_SUM:
|
|
s += v;
|
|
newScalars->SetTuple(idx,&s);
|
|
break;
|
|
}
|
|
}//not first visit
|
|
}
|
|
|
|
const char *vtkGaussianSplatter::GetAccumulationModeAsString()
|
|
{
|
|
if ( this->AccumulationMode == VTK_ACCUMULATION_MODE_MIN )
|
|
{
|
|
return "Minimum";
|
|
}
|
|
else if ( this->AccumulationMode == VTK_ACCUMULATION_MODE_MAX )
|
|
{
|
|
return "Maximum";
|
|
}
|
|
else //if ( this->AccumulationMode == VTK_ACCUMULATION_MODE_SUM )
|
|
{
|
|
return "Sum";
|
|
}
|
|
}
|
|
|
|
void vtkGaussianSplatter::PrintSelf(ostream& os, vtkIndent indent)
|
|
{
|
|
this->Superclass::PrintSelf(os,indent);
|
|
|
|
os << indent << "Sample Dimensions: ("
|
|
<< this->SampleDimensions[0] << ", "
|
|
<< this->SampleDimensions[1] << ", "
|
|
<< this->SampleDimensions[2] << ")\n";
|
|
|
|
os << indent << "Radius: " << this->Radius << "\n";
|
|
os << indent << "Exponent Factor: " << this->ExponentFactor << "\n";
|
|
|
|
os << indent << "ModelBounds: \n";
|
|
os << indent << " Xmin,Xmax: (" << this->ModelBounds[0]
|
|
<< ", " << this->ModelBounds[1] << ")\n";
|
|
os << indent << " Ymin,Ymax: (" << this->ModelBounds[2]
|
|
<< ", " << this->ModelBounds[3] << ")\n";
|
|
os << indent << " Zmin,Zmax: (" << this->ModelBounds[4]
|
|
<< ", " << this->ModelBounds[5] << ")\n";
|
|
|
|
os << indent << "Normal Warping: "
|
|
<< (this->NormalWarping ? "On\n" : "Off\n");
|
|
os << indent << "Eccentricity: " << this->Eccentricity << "\n";
|
|
|
|
os << indent << "Scalar Warping: "
|
|
<< (this->ScalarWarping ? "On\n" : "Off\n");
|
|
os << indent << "Scale Factor: " << this->ScaleFactor << "\n";
|
|
|
|
os << indent << "Capping: " << (this->Capping ? "On\n" : "Off\n");
|
|
os << indent << "Cap Value: " << this->CapValue << "\n";
|
|
|
|
os << indent << "Accumulation Mode: "
|
|
<< this->GetAccumulationModeAsString() << "\n";
|
|
|
|
os << indent << "Null Value: " << this->NullValue << "\n";
|
|
}
|
|
|
|
int vtkGaussianSplatter::FillInputPortInformation(
|
|
int vtkNotUsed(port), vtkInformation* info)
|
|
{
|
|
info->Set(vtkAlgorithm::INPUT_REQUIRED_DATA_TYPE(), "vtkDataSet");
|
|
return 1;
|
|
}
|
|
|