Cloned library of VTK-5.0.0 with extra build files for internal package management.
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.

666 lines
22 KiB

2 years ago
/*=========================================================================
Program: Visualization Toolkit
Module: $RCSfile: vtkFixedPointVolumeRayCastMIPHelper.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 "vtkFixedPointVolumeRayCastMIPHelper.h"
#include "vtkImageData.h"
#include "vtkCommand.h"
#include "vtkFixedPointVolumeRayCastMapper.h"
#include "vtkObjectFactory.h"
#include "vtkRenderWindow.h"
#include "vtkVolume.h"
#include "vtkVolumeProperty.h"
#include "vtkFixedPointRayCastImage.h"
#include <math.h>
vtkCxxRevisionMacro(vtkFixedPointVolumeRayCastMIPHelper, "$Revision: 1.6 $");
vtkStandardNewMacro(vtkFixedPointVolumeRayCastMIPHelper);
// Construct a new vtkFixedPointVolumeRayCastMIPHelper with default values
vtkFixedPointVolumeRayCastMIPHelper::vtkFixedPointVolumeRayCastMIPHelper()
{
}
// Destruct a vtkFixedPointVolumeRayCastMIPHelper - clean up any memory used
vtkFixedPointVolumeRayCastMIPHelper::~vtkFixedPointVolumeRayCastMIPHelper()
{
}
// This method is called when the interpolation type is nearest neighbor and
// the data contains one component. In the inner loop we will compute the
// maximum value (in native type). After we have a maximum value for the ray
// we will convert it to unsigned short using the scale/shift, then use this
// index to lookup the final color/opacity.
template <class T>
void vtkFixedPointMIPHelperGenerateImageOneNN( T *data,
int threadID,
int threadCount,
vtkFixedPointVolumeRayCastMapper *mapper,
vtkVolume *vtkNotUsed(vol))
{
VTKKWRCHelper_InitializationAndLoopStartNN();
VTKKWRCHelper_InitializeMIPOneNN();
VTKKWRCHelper_SpaceLeapSetup();
if ( cropping )
{
int maxValueDefined = 0;
unsigned short maxIdx = 0;
for ( k = 0; k < numSteps; k++ )
{
if ( k )
{
mapper->FixedPointIncrement( pos, dir );
}
VTKKWRCHelper_MIPSpaceLeapCheck( maxIdx, maxValueDefined );
if ( !mapper->CheckIfCropped( pos ) )
{
mapper->ShiftVectorDown( pos, spos );
dptr = data + spos[0]*inc[0] + spos[1]*inc[1] + spos[2]*inc[2];
if ( !maxValueDefined || *dptr > maxValue )
{
maxValue = *dptr;
maxIdx = static_cast<unsigned short>((maxValue + shift[0])*scale[0]);
maxValueDefined = 1;
}
}
}
if ( maxValueDefined )
{
VTKKWRCHelper_LookupColorMax( colorTable[0], scalarOpacityTable[0], maxIdx, imagePtr );
}
else
{
imagePtr[0] = imagePtr[1] = imagePtr[2] = imagePtr[3] = 0;
}
}
else
{
unsigned short maxIdx =
static_cast<unsigned short>((maxValue + shift[0])*scale[0]);
for ( k = 0; k < numSteps; k++ )
{
if ( k )
{
mapper->FixedPointIncrement( pos, dir );
}
VTKKWRCHelper_MIPSpaceLeapCheck( maxIdx, 1 );
mapper->ShiftVectorDown( pos, spos );
dptr = data + spos[0]*inc[0] + spos[1]*inc[1] + spos[2]*inc[2];
maxValue = ( *dptr > maxValue )?(*dptr):(maxValue);
maxIdx = static_cast<unsigned short>((maxValue + shift[0])*scale[0]);
}
VTKKWRCHelper_LookupColorMax( colorTable[0], scalarOpacityTable[0], maxIdx, imagePtr );
}
VTKKWRCHelper_IncrementAndLoopEnd();
}
// This method is called when the interpolation type is nearest neighbor and
// the data has two or four dependent components. If it is four, they must be
// unsigned char components. Compute max of last components in native type,
// then use first component to look up a color (2 component data) or first three
// as the color directly (four component data). Lookup alpha off the last component.
template <class T>
void vtkFixedPointMIPHelperGenerateImageDependentNN( T *data,
int threadID,
int threadCount,
vtkFixedPointVolumeRayCastMapper *mapper,
vtkVolume *vtkNotUsed(vol))
{
VTKKWRCHelper_InitializationAndLoopStartNN();
VTKKWRCHelper_InitializeMIPMultiNN();
VTKKWRCHelper_SpaceLeapSetup();
int maxValueDefined = 0;
unsigned short maxIdxS = 0;
for ( k = 0; k < numSteps; k++ )
{
if ( k )
{
mapper->FixedPointIncrement( pos, dir );
}
VTKKWRCHelper_MIPSpaceLeapCheck( maxIdxS, maxValueDefined );
VTKKWRCHelper_CroppingCheckNN( pos );
mapper->ShiftVectorDown( pos, spos );
dptr = data + spos[0]*inc[0] + spos[1]*inc[1] + spos[2]*inc[2];
if ( !maxValueDefined || *(dptr + components - 1) > maxValue[components-1] )
{
for ( c = 0; c < components; c++ )
{
maxValue[c] = *(dptr+c);
}
maxIdxS = (unsigned short)((maxValue[components-1] +
shift[components-1])*scale[components-1]);
maxValueDefined = 1;
}
}
if ( maxValueDefined )
{
unsigned short maxIdx[4];
for ( c = 0; c < components; c++ )
{
maxIdx[c] = (unsigned short)((maxValue[c] + shift[c])*scale[c]);
}
VTKKWRCHelper_LookupDependentColorUS( colorTable[0], scalarOpacityTable[0],
maxIdx, components, imagePtr );
}
else
{
imagePtr[0] = imagePtr[1] = imagePtr[2] = imagePtr[3] = 0;
}
VTKKWRCHelper_IncrementAndLoopEnd();
}
// This method is called when the interpolation type is nearest neighbor and
// the data has more than one independent components. We compute the max of
// each component along the ray in native type, then use the scale/shift to
// convert this into an unsigned short index value. We use the index values
// to lookup the color/opacity per component, then use the component weights to
// blend these into one final color.
template <class T>
void vtkFixedPointMIPHelperGenerateImageIndependentNN( T *data,
int threadID,
int threadCount,
vtkFixedPointVolumeRayCastMapper *mapper,
vtkVolume *vol)
{
VTKKWRCHelper_InitializeWeights();
VTKKWRCHelper_InitializationAndLoopStartNN();
VTKKWRCHelper_InitializeMIPMultiNN();
VTKKWRCHelper_SpaceLeapSetupMulti();
int maxValueDefined = 0;
unsigned short maxIdx[4];
for ( k = 0; k < numSteps; k++ )
{
if ( k )
{
mapper->FixedPointIncrement( pos, dir );
}
VTKKWRCHelper_CroppingCheckNN( pos );
VTKKWRCHelper_MIPSpaceLeapPopulateMulti( maxIdx )
mapper->ShiftVectorDown( pos, spos );
dptr = data + spos[0]*inc[0] + spos[1]*inc[1] + spos[2]*inc[2];
if ( !maxValueDefined )
{
for ( c = 0; c < components; c++ )
{
maxValue[c] = *(dptr+c);
maxIdx[c] = (unsigned short)((maxValue[c] + shift[c])*scale[c]);
}
maxValueDefined = 1;
}
else
{
for ( c = 0; c < components; c++ )
{
if ( VTKKWRCHelper_MIPSpaceLeapCheckMulti( c ) &&
*(dptr + c) > maxValue[c] )
{
maxValue[c] = *(dptr+c);
maxIdx[c] = (unsigned short)((maxValue[c] + shift[c])*scale[c]);
}
}
}
}
imagePtr[0] = imagePtr[1] = imagePtr[2] = imagePtr[3] = 0;
if ( maxValueDefined )
{
VTKKWRCHelper_LookupAndCombineIndependentColorsMax( colorTable,
scalarOpacityTable,
maxIdx, weights,
components, imagePtr );
}
VTKKWRCHelper_IncrementAndLoopEnd();
}
// This method is called when the interpolation type is linear, the
// data contains one component and scale = 1.0 and shift = 0.0. This is
// the simple case were we do not need to apply scale/shift in the
// inner loop. In the inner loop we compute the eight cell vertex values
// (if we have changed cells). We compute our weights within the cell
// according to our fractional position within the cell, and apply trilinear
// interpolation to compute the index. We find the maximum index along
// the ray, and then use this to look up a final color.
template <class T>
void vtkFixedPointMIPHelperGenerateImageOneSimpleTrilin( T *dataPtr,
int threadID,
int threadCount,
vtkFixedPointVolumeRayCastMapper *mapper,
vtkVolume *vtkNotUsed(vol))
{
VTKKWRCHelper_InitializationAndLoopStartTrilin();
VTKKWRCHelper_InitializeMIPOneTrilin();
VTKKWRCHelper_SpaceLeapSetup();
int maxValueDefined = 0;
unsigned short maxIdx=0;
unsigned short maxScalar = 0;
for ( k = 0; k < numSteps; k++ )
{
if ( k )
{
mapper->FixedPointIncrement( pos, dir );
}
VTKKWRCHelper_MIPSpaceLeapCheck( maxIdx, maxValueDefined );
VTKKWRCHelper_CroppingCheckTrilin( pos );
mapper->ShiftVectorDown( pos, spos );
if ( spos[0] != oldSPos[0] ||
spos[1] != oldSPos[1] ||
spos[2] != oldSPos[2] )
{
oldSPos[0] = spos[0];
oldSPos[1] = spos[1];
oldSPos[2] = spos[2];
dptr = dataPtr + spos[0]*inc[0] + spos[1]*inc[1] + spos[2]*inc[2];
VTKKWRCHelper_GetCellScalarValuesSimple( dptr );
maxScalar = (A>B)?(A):(B);
maxScalar = (C>maxScalar)?(C):(maxScalar);
maxScalar = (D>maxScalar)?(D):(maxScalar);
maxScalar = (E>maxScalar)?(E):(maxScalar);
maxScalar = (F>maxScalar)?(F):(maxScalar);
maxScalar = (G>maxScalar)?(G):(maxScalar);
maxScalar = (H>maxScalar)?(H):(maxScalar);
}
if ( !maxValueDefined || maxScalar > maxValue )
{
VTKKWRCHelper_ComputeWeights(pos);
VTKKWRCHelper_InterpolateScalar(val);
if ( !maxValueDefined || val > maxValue )
{
maxValue = val;
maxIdx = static_cast<unsigned short>(maxValue);
maxValueDefined = 1;
}
}
}
if ( maxValueDefined )
{
VTKKWRCHelper_LookupColorMax( colorTable[0], scalarOpacityTable[0], maxIdx, imagePtr );
}
else
{
imagePtr[0] = imagePtr[1] = imagePtr[2] = imagePtr[3] = 0;
}
VTKKWRCHelper_IncrementAndLoopEnd();
}
// This method is called when the interpolation type is linear, the
// data contains one component and scale != 1.0 or shift != 0.0. This
// means that we need to apply scale/shift in the inner loop to compute
// an unsigned short index value. In the inner loop we compute the eight cell
// vertex values (as unsigned short indices, if we have changed cells). We
// compute our weights within the cell according to our fractional position
// within the cell, and apply trilinear interpolation to compute the index.
// We find the maximum index along the ray, and then use this to look up a
// final color.
template <class T>
void vtkFixedPointMIPHelperGenerateImageOneTrilin( T *dataPtr,
int threadID,
int threadCount,
vtkFixedPointVolumeRayCastMapper *mapper,
vtkVolume *vtkNotUsed(vol))
{
VTKKWRCHelper_InitializationAndLoopStartTrilin();
VTKKWRCHelper_InitializeMIPOneTrilin();
VTKKWRCHelper_SpaceLeapSetup();
int maxValueDefined = 0;
unsigned short maxIdx = 0;
for ( k = 0; k < numSteps; k++ )
{
if ( k )
{
mapper->FixedPointIncrement( pos, dir );
}
VTKKWRCHelper_CroppingCheckTrilin( pos );
VTKKWRCHelper_MIPSpaceLeapCheck( maxIdx, maxValueDefined );
mapper->ShiftVectorDown( pos, spos );
if ( spos[0] != oldSPos[0] ||
spos[1] != oldSPos[1] ||
spos[2] != oldSPos[2] )
{
oldSPos[0] = spos[0];
oldSPos[1] = spos[1];
oldSPos[2] = spos[2];
dptr = dataPtr + spos[0]*inc[0] + spos[1]*inc[1] + spos[2]*inc[2];
VTKKWRCHelper_GetCellScalarValues( dptr, scale[0], shift[0] );
}
VTKKWRCHelper_ComputeWeights(pos);
VTKKWRCHelper_InterpolateScalar(val);
if ( !maxValueDefined || val > maxValue )
{
maxValue = val;
maxIdx = static_cast<unsigned short>(maxValue);
maxValueDefined = 1;
}
}
if ( maxValueDefined )
{
VTKKWRCHelper_LookupColorMax( colorTable[0], scalarOpacityTable[0], maxIdx, imagePtr );
}
else
{
imagePtr[0] = imagePtr[1] = imagePtr[2] = imagePtr[3] = 0;
}
VTKKWRCHelper_IncrementAndLoopEnd();
}
// This method is used when the interpolation type is linear, the data has
// two or four components and the components are not considered independent.
// For four component d>>(VTKKW_FP_SHIFT - 8));ata, the data must be unsigned char in type. In the
// inner loop we get the data value for the eight cell corners (if we have
// changed cells) for all components as unsigned shorts (we use the
// scale/shift to ensure the correct range). We compute our weights within
// the cell according to our fractional position within the cell, and apply
// trilinear interpolation to compute the index values. For two component data,
// We use the first index to lookup a color and the second to look up an opacity
// for this sample. For four component data we use the first three components
// directly as a color, then we look up the opacity using the fourth component.
// We then composite this into the color computed so far along the ray, and
// check if we can terminate at this point (if the accumulated opacity is
// higher than some threshold).
template <class T>
void vtkFixedPointMIPHelperGenerateImageDependentTrilin( T *dataPtr,
int threadID,
int threadCount,
vtkFixedPointVolumeRayCastMapper *mapper,
vtkVolume *vtkNotUsed(vol))
{
VTKKWRCHelper_InitializationAndLoopStartTrilin();
VTKKWRCHelper_InitializeMIPMultiTrilin();
VTKKWRCHelper_SpaceLeapSetup();
int maxValueDefined = 0;
unsigned short maxIdx = 0;
for ( k = 0; k < numSteps; k++ )
{
if ( k )
{
mapper->FixedPointIncrement( pos, dir );
}
VTKKWRCHelper_CroppingCheckTrilin( pos );
VTKKWRCHelper_MIPSpaceLeapCheck( maxIdx, maxValueDefined );
mapper->ShiftVectorDown( pos, spos );
if ( spos[0] != oldSPos[0] ||
spos[1] != oldSPos[1] ||
spos[2] != oldSPos[2] )
{
oldSPos[0] = spos[0];
oldSPos[1] = spos[1];
oldSPos[2] = spos[2];
for ( c= 0; c < components; c++ )
{
dptr = dataPtr + spos[0]*inc[0] + spos[1]*inc[1] + spos[2]*inc[2] + c;
VTKKWRCHelper_GetCellComponentScalarValues( dptr, c, scale[c], shift[c] );
}
}
VTKKWRCHelper_ComputeWeights(pos);
VTKKWRCHelper_InterpolateScalarComponent( val, c, components );
if ( !maxValueDefined || (val[components-1] > maxValue[components-1]) )
{
for ( c= 0; c < components; c++ )
{
maxValue[c] = val[c];
}
maxIdx = static_cast<unsigned short>((maxValue[components-1] +
shift[components-1])*scale[components-1]);
maxValueDefined = 1;
}
}
if ( maxValueDefined )
{
VTKKWRCHelper_LookupDependentColorUS( colorTable[0],
scalarOpacityTable[0],
maxValue, components, imagePtr );
}
else
{
imagePtr[0] = imagePtr[1] = imagePtr[2] = imagePtr[3] = 0;
}
VTKKWRCHelper_IncrementAndLoopEnd();
}
// This method is used when the interpolation type is linear, the data has
// more than one component and the components are considered independent. In
// the inner loop we get the data value for the eight cell corners (if we have
// changed cells) for all components as an unsigned shorts (we have to use the
// scale/shift to ensure that we obtained unsigned short indices) We compute our
// weights within the cell according to our fractional position within the cell,
// and apply trilinear interpolation to compute a value for each component. We do
// this for each sample along the ray to find a maximum value per component, then
// we look up a color/opacity for each component and blend them according to the
// component weights.
template <class T>
void vtkFixedPointMIPHelperGenerateImageIndependentTrilin( T *dataPtr,
int threadID,
int threadCount,
vtkFixedPointVolumeRayCastMapper *mapper,
vtkVolume *vol)
{
VTKKWRCHelper_InitializeWeights();
VTKKWRCHelper_InitializationAndLoopStartTrilin();
VTKKWRCHelper_InitializeMIPMultiTrilin();
int maxValueDefined = 0;
for ( k = 0; k < numSteps; k++ )
{
if ( k )
{
mapper->FixedPointIncrement( pos, dir );
}
VTKKWRCHelper_CroppingCheckTrilin( pos );
mapper->ShiftVectorDown( pos, spos );
if ( spos[0] != oldSPos[0] ||
spos[1] != oldSPos[1] ||
spos[2] != oldSPos[2] )
{
oldSPos[0] = spos[0];
oldSPos[1] = spos[1];
oldSPos[2] = spos[2];
for ( c= 0; c < components; c++ )
{
dptr = dataPtr + spos[0]*inc[0] + spos[1]*inc[1] + spos[2]*inc[2] + c;
VTKKWRCHelper_GetCellComponentScalarValues( dptr, c, scale[c], shift[c] );
}
}
VTKKWRCHelper_ComputeWeights(pos);
VTKKWRCHelper_InterpolateScalarComponent( val, c, components );
if ( !maxValueDefined )
{
for ( c= 0; c < components; c++ )
{
maxValue[c] = val[c];
}
maxValueDefined = 1;
}
else
{
for ( c= 0; c < components; c++ )
{
if ( (val[c] > maxValue[c]) )
{
maxValue[c] = val[c];
}
}
}
}
imagePtr[0] = imagePtr[1] = imagePtr[2] = imagePtr[3] = 0;
if ( maxValueDefined )
{
VTKKWRCHelper_LookupAndCombineIndependentColorsMax( colorTable, scalarOpacityTable,
maxValue, weights, components, imagePtr );
}
VTKKWRCHelper_IncrementAndLoopEnd();
}
void vtkFixedPointVolumeRayCastMIPHelper::GenerateImage( int threadID,
int threadCount,
vtkVolume *vol,
vtkFixedPointVolumeRayCastMapper *mapper )
{
void *dataPtr = mapper->GetInput()->GetScalarPointer();
int scalarType = mapper->GetInput()->GetScalarType();
// Nearest Neighbor interpolate
if ( mapper->ShouldUseNearestNeighborInterpolation( vol ) )
{
// One component data
if ( mapper->GetInput()->GetNumberOfScalarComponents() == 1 )
{
switch ( scalarType )
{
vtkTemplateMacro(
vtkFixedPointMIPHelperGenerateImageOneNN(
(VTK_TT *)(dataPtr),
threadID, threadCount, mapper, vol) );
}
}
// More that one independent components
else if ( vol->GetProperty()->GetIndependentComponents() )
{
switch ( scalarType )
{
vtkTemplateMacro(
vtkFixedPointMIPHelperGenerateImageIndependentNN(
(VTK_TT *)(dataPtr),
threadID, threadCount, mapper, vol) );
}
}
// Dependent (color) components
else
{
switch ( scalarType )
{
vtkTemplateMacro(
vtkFixedPointMIPHelperGenerateImageDependentNN(
(VTK_TT *)(dataPtr),
threadID, threadCount, mapper, vol) );
}
}
}
// Trilinear Interpolation
else
{
// One component
if ( mapper->GetInput()->GetNumberOfScalarComponents() == 1 )
{
// Scale == 1.0 and shift == 0.0 - simple case (faster)
if ( mapper->GetTableScale()[0] == 1.0 && mapper->GetTableShift()[0] == 0.0 )
{
switch ( scalarType )
{
vtkTemplateMacro(
vtkFixedPointMIPHelperGenerateImageOneSimpleTrilin(
(VTK_TT *)(dataPtr),
threadID, threadCount, mapper, vol) );
}
}
// Scale != 1.0 or shift != 0.0 - must apply scale/shift in inner loop
else
{
switch ( scalarType )
{
vtkTemplateMacro(
vtkFixedPointMIPHelperGenerateImageOneTrilin(
(VTK_TT *)(dataPtr),
threadID, threadCount, mapper, vol) );
}
}
}
// Indepedent components (more than one)
else if ( vol->GetProperty()->GetIndependentComponents() )
{
switch ( scalarType )
{
vtkTemplateMacro(
vtkFixedPointMIPHelperGenerateImageIndependentTrilin(
(VTK_TT *)(dataPtr),
threadID, threadCount, mapper, vol) );
}
}
// Dependent components
else
{
switch ( scalarType )
{
vtkTemplateMacro(
vtkFixedPointMIPHelperGenerateImageDependentTrilin(
(VTK_TT *)(dataPtr),
threadID, threadCount, mapper, vol) );
}
}
}
}
// Print method for vtkFixedPointVolumeRayCastMIPHelper
void vtkFixedPointVolumeRayCastMIPHelper::PrintSelf(ostream& os, vtkIndent indent)
{
this->Superclass::PrintSelf(os,indent);
}