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.
 
 
 
 
 
 

832 lines
26 KiB

/*=========================================================================
Program: Visualization Toolkit
Module: $RCSfile: vtkUnstructuredGridVolumeRayCastMapper.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 "vtkUnstructuredGridVolumeRayCastMapper.h"
#include "vtkCamera.h"
#include "vtkEncodedGradientEstimator.h"
#include "vtkEncodedGradientShader.h"
#include "vtkFiniteDifferenceGradientEstimator.h"
#include "vtkUnstructuredGrid.h"
#include "vtkMath.h"
#include "vtkMultiThreader.h"
#include "vtkObjectFactory.h"
#include "vtkPlaneCollection.h"
#include "vtkPointData.h"
#include "vtkRenderWindow.h"
#include "vtkRenderer.h"
#include "vtkTimerLog.h"
#include "vtkTransform.h"
#include "vtkVolumeProperty.h"
#include "vtkUnstructuredGridBunykRayCastFunction.h"
#include "vtkUnstructuredGridVolumeRayCastIterator.h"
#include "vtkUnstructuredGridPreIntegration.h"
#include "vtkUnstructuredGridPartialPreIntegration.h"
#include "vtkUnstructuredGridHomogeneousRayIntegrator.h"
#include "vtkRayCastImageDisplayHelper.h"
#include "vtkDoubleArray.h"
#include "vtkIdList.h"
#include <math.h>
VTK_THREAD_RETURN_TYPE UnstructuredGridVolumeRayCastMapper_CastRays( void *arg );
vtkCxxRevisionMacro(vtkUnstructuredGridVolumeRayCastMapper, "$Revision: 1.2 $");
vtkStandardNewMacro(vtkUnstructuredGridVolumeRayCastMapper);
vtkCxxSetObjectMacro(vtkUnstructuredGridVolumeRayCastMapper, RayCastFunction,
vtkUnstructuredGridVolumeRayCastFunction);
vtkCxxSetObjectMacro(vtkUnstructuredGridVolumeRayCastMapper, RayIntegrator,
vtkUnstructuredGridVolumeRayIntegrator);
// Construct a new vtkUnstructuredGridVolumeRayCastMapper with default values
vtkUnstructuredGridVolumeRayCastMapper::vtkUnstructuredGridVolumeRayCastMapper()
{
this->ImageSampleDistance = 1.0;
this->MinimumImageSampleDistance = 1.0;
this->MaximumImageSampleDistance = 10.0;
this->AutoAdjustSampleDistances = 1;
this->ImageMemorySize[0] = 0;
this->ImageMemorySize[1] = 0;
this->Threader = vtkMultiThreader::New();
this->NumberOfThreads = this->Threader->GetNumberOfThreads();
this->Image = NULL;
this->RenderTimeTable = NULL;
this->RenderVolumeTable = NULL;
this->RenderRendererTable = NULL;
this->RenderTableSize = 0;
this->RenderTableEntries = 0;
this->ZBuffer = NULL;
this->ZBufferSize[0] = 0;
this->ZBufferSize[1] = 0;
this->ZBufferOrigin[0] = 0;
this->ZBufferOrigin[1] = 0;
this->IntermixIntersectingGeometry = 1;
this->ImageDisplayHelper = vtkRayCastImageDisplayHelper::New();
this->RayCastFunction = vtkUnstructuredGridBunykRayCastFunction::New();
this->RayIntegrator = NULL;
this->RealRayIntegrator = NULL;
this->ScalarMode = VTK_SCALAR_MODE_DEFAULT;
this->ArrayName = new char[1];
this->ArrayName[0] = '\0';
this->ArrayId = -1;
this->ArrayAccessMode = VTK_GET_ARRAY_BY_ID;
}
// Destruct a vtkUnstructuredGridVolumeRayCastMapper - clean up any memory used
vtkUnstructuredGridVolumeRayCastMapper::~vtkUnstructuredGridVolumeRayCastMapper()
{
this->Threader->Delete();
if ( this->Image )
{
delete [] this->Image;
}
if ( this->RenderTableSize )
{
delete [] this->RenderTimeTable;
delete [] this->RenderVolumeTable;
delete [] this->RenderRendererTable;
}
this->ImageDisplayHelper->Delete();
this->SetRayCastFunction(NULL);
this->SetRayIntegrator(NULL);
if (this->RealRayIntegrator)
{
this->RealRayIntegrator->UnRegister(this);
}
delete[] this->ArrayName;
}
float vtkUnstructuredGridVolumeRayCastMapper::RetrieveRenderTime( vtkRenderer *ren,
vtkVolume *vol )
{
int i;
for ( i = 0; i < this->RenderTableEntries; i++ )
{
if ( this->RenderVolumeTable[i] == vol &&
this->RenderRendererTable[i] == ren )
{
return this->RenderTimeTable[i];
}
}
return 0.0;
}
void vtkUnstructuredGridVolumeRayCastMapper::StoreRenderTime( vtkRenderer *ren,
vtkVolume *vol,
float time )
{
int i;
for ( i = 0; i < this->RenderTableEntries; i++ )
{
if ( this->RenderVolumeTable[i] == vol &&
this->RenderRendererTable[i] == ren )
{
this->RenderTimeTable[i] = time;
return;
}
}
// Need to increase size
if ( this->RenderTableEntries >= this->RenderTableSize )
{
if ( this->RenderTableSize == 0 )
{
this->RenderTableSize = 10;
}
else
{
this->RenderTableSize *= 2;
}
float *oldTimePtr = this->RenderTimeTable;
vtkVolume **oldVolumePtr = this->RenderVolumeTable;
vtkRenderer **oldRendererPtr = this->RenderRendererTable;
this->RenderTimeTable = new float [this->RenderTableSize];
this->RenderVolumeTable = new vtkVolume *[this->RenderTableSize];
this->RenderRendererTable = new vtkRenderer *[this->RenderTableSize];
for (i = 0; i < this->RenderTableEntries; i++ )
{
this->RenderTimeTable[i] = oldTimePtr[i];
this->RenderVolumeTable[i] = oldVolumePtr[i];
this->RenderRendererTable[i] = oldRendererPtr[i];
}
delete [] oldTimePtr;
delete [] oldVolumePtr;
delete [] oldRendererPtr;
}
this->RenderTimeTable[this->RenderTableEntries] = time;
this->RenderVolumeTable[this->RenderTableEntries] = vol;
this->RenderRendererTable[this->RenderTableEntries] = ren;
this->RenderTableEntries++;
}
void vtkUnstructuredGridVolumeRayCastMapper::SelectScalarArray(int arrayNum)
{
if ( (this->ArrayId == arrayNum)
&& (this->ArrayAccessMode == VTK_GET_ARRAY_BY_ID) )
{
return;
}
this->Modified();
this->ArrayId = arrayNum;
this->ArrayAccessMode = VTK_GET_ARRAY_BY_ID;
}
void vtkUnstructuredGridVolumeRayCastMapper::SelectScalarArray(const char *arrayName)
{
if ( !arrayName
|| ( (strcmp(this->ArrayName, arrayName) == 0)
&& (this->ArrayAccessMode == VTK_GET_ARRAY_BY_ID) ) )
{
return;
}
this->Modified();
delete[] this->ArrayName;
this->ArrayName = new char[strlen(arrayName) + 1];
strcpy(this->ArrayName, arrayName);
this->ArrayAccessMode = VTK_GET_ARRAY_BY_NAME;
}
// Return the method for obtaining scalar data.
const char *vtkUnstructuredGridVolumeRayCastMapper::GetScalarModeAsString(void)
{
if ( this->ScalarMode == VTK_SCALAR_MODE_USE_CELL_DATA )
{
return "UseCellData";
}
else if ( this->ScalarMode == VTK_SCALAR_MODE_USE_POINT_DATA )
{
return "UsePointData";
}
else if ( this->ScalarMode == VTK_SCALAR_MODE_USE_POINT_FIELD_DATA )
{
return "UsePointFieldData";
}
else if ( this->ScalarMode == VTK_SCALAR_MODE_USE_CELL_FIELD_DATA )
{
return "UseCellFieldData";
}
else
{
return "Default";
}
}
void vtkUnstructuredGridVolumeRayCastMapper::ReleaseGraphicsResources(vtkWindow *)
{
}
void vtkUnstructuredGridVolumeRayCastMapper::Render( vtkRenderer *ren, vtkVolume *vol )
{
int i;
// Check for input
if ( this->GetInput() == NULL )
{
vtkErrorMacro(<< "No Input!");
return;
}
this->Scalars = this->GetScalars(this->GetInput(), this->ScalarMode,
this->ArrayAccessMode,
this->ArrayId, this->ArrayName,
this->CellScalars);
if (this->Scalars == NULL)
{
vtkErrorMacro("Can't use the ray cast mapper without scalars!");
return;
}
this->GetInput()->UpdateInformation();
this->GetInput()->SetUpdateExtentToWholeExtent();
this->GetInput()->Update();
// Check to make sure we have an appropriate integrator.
if (this->RayIntegrator)
{
if (this->RealRayIntegrator != this->RayIntegrator)
{
if (this->RealRayIntegrator)
{
this->RealRayIntegrator->UnRegister(this);
}
this->RealRayIntegrator = this->RayIntegrator;
this->RealRayIntegrator->Register(this);
}
}
else
{
#define ESTABLISH_INTEGRATOR(classname) \
if ( !this->RealRayIntegrator \
|| (!this->RealRayIntegrator->IsA(#classname)) ) \
{ \
if (this->RealRayIntegrator) this->RealRayIntegrator->UnRegister(this); \
this->RealRayIntegrator = classname::New(); \
this->RealRayIntegrator->Register(this); \
this->RealRayIntegrator->Delete(); \
} \
if (this->CellScalars)
{
ESTABLISH_INTEGRATOR(vtkUnstructuredGridHomogeneousRayIntegrator);
}
else
{
if (vol->GetProperty()->GetIndependentComponents())
{
ESTABLISH_INTEGRATOR(vtkUnstructuredGridPreIntegration);
}
else
{
ESTABLISH_INTEGRATOR(vtkUnstructuredGridPartialPreIntegration);
}
}
}
#undef ESTABLISH_INTEGRATOR
// Start timing now. We didn't want to capture the update of the
// input data in the times
this->Timer->StartTimer();
int oldImageMemorySize[2];
oldImageMemorySize[0] = this->ImageMemorySize[0];
oldImageMemorySize[1] = this->ImageMemorySize[1];
// If we are automatically adjusting the size to achieve a desired frame
// rate, then do that adjustment here. Base the new image sample distance
// on the previous one and the previous render time. Don't let
// the adjusted image sample distance be less than the minimum image sample
// distance or more than the maximum image sample distance.
float oldImageSampleDistance = this->ImageSampleDistance;
if ( this->AutoAdjustSampleDistances )
{
float oldTime = this->RetrieveRenderTime( ren, vol );
float newTime = vol->GetAllocatedRenderTime();
this->ImageSampleDistance *= sqrt(oldTime / newTime);
this->ImageSampleDistance =
(this->ImageSampleDistance>this->MaximumImageSampleDistance)?
(this->MaximumImageSampleDistance):(this->ImageSampleDistance);
this->ImageSampleDistance =
(this->ImageSampleDistance<this->MinimumImageSampleDistance)?
(this->MinimumImageSampleDistance):(this->ImageSampleDistance);
}
// The full image fills the viewport. First, compute the actual viewport
// size, then divide by the ImageSampleDistance to find the full image
// size in pixels
int width, height;
ren->GetTiledSize(&width, &height);
this->ImageViewportSize[0] =
static_cast<int>(width/this->ImageSampleDistance);
this->ImageViewportSize[1] =
static_cast<int>(height/this->ImageSampleDistance);
this->ImageInUseSize[0] = this->ImageViewportSize[0];
this->ImageInUseSize[1] = this->ImageViewportSize[1];
this->ImageOrigin[0] = 0;
this->ImageOrigin[1] = 0;
// What is a power of 2 size big enough to fit this image?
this->ImageMemorySize[0] = 32;
this->ImageMemorySize[1] = 32;
while ( this->ImageMemorySize[0] < this->ImageInUseSize[0] )
{
this->ImageMemorySize[0] *= 2;
}
while ( this->ImageMemorySize[1] < this->ImageInUseSize[1] )
{
this->ImageMemorySize[1] *= 2;
}
// If the old image size is much too big (more than twice in
// either direction) then set the old width to 0 which will
// cause the image to be recreated
if ( oldImageMemorySize[0] > 2*this->ImageMemorySize[0] ||
oldImageMemorySize[1] > 2*this->ImageMemorySize[1] )
{
oldImageMemorySize[0] = 0;
}
// If the old image is big enough (but not too big - we handled
// that above) then we'll bump up our required size to the
// previous one. This will keep us from thrashing.
if ( oldImageMemorySize[0] >= this->ImageMemorySize[0] &&
oldImageMemorySize[1] >= this->ImageMemorySize[1] )
{
this->ImageMemorySize[0] = oldImageMemorySize[0];
this->ImageMemorySize[1] = oldImageMemorySize[1];
}
// Do we already have a texture big enough? If not, create a new one and
// clear it.
if ( !this->Image ||
this->ImageMemorySize[0] > oldImageMemorySize[0] ||
this->ImageMemorySize[1] > oldImageMemorySize[1] )
{
// If there is an image there must be row bounds
if ( this->Image )
{
delete [] this->Image;
}
this->Image = new unsigned char[(this->ImageMemorySize[0] *
this->ImageMemorySize[1] * 4)];
unsigned char *ucptr = this->Image;
for ( i = 0; i < this->ImageMemorySize[0]*this->ImageMemorySize[1]; i++ )
{
*(ucptr++) = 0;
*(ucptr++) = 0;
*(ucptr++) = 0;
*(ucptr++) = 0;
}
}
// Capture the zbuffer if necessary
if ( this->IntermixIntersectingGeometry &&
ren->GetNumberOfPropsRendered() )
{
int x1, x2, y1, y2;
double *viewport = ren->GetViewport();
int *renWinSize = ren->GetRenderWindow()->GetSize();
// turn this->ImageOrigin into (x1,y1) in window (not viewport!)
// coordinates.
x1 = static_cast<int> (
viewport[0] * static_cast<float>(renWinSize[0]) +
static_cast<float>(this->ImageOrigin[0]) * this->ImageSampleDistance );
y1 = static_cast<int> (
viewport[1] * static_cast<float>(renWinSize[1]) +
static_cast<float>(this->ImageOrigin[1]) * this->ImageSampleDistance);
// compute z buffer size
this->ZBufferSize[0] = static_cast<int>(
static_cast<float>(this->ImageInUseSize[0]) * this->ImageSampleDistance);
this->ZBufferSize[1] = static_cast<int>(
static_cast<float>(this->ImageInUseSize[1]) * this->ImageSampleDistance);
// Use the size to compute (x2,y2) in window coordinates
x2 = x1 + this->ZBufferSize[0] - 1;
y2 = y1 + this->ZBufferSize[1] - 1;
// This is the z buffer origin (in viewport coordinates)
this->ZBufferOrigin[0] = static_cast<int>(
static_cast<float>(this->ImageOrigin[0]) * this->ImageSampleDistance);
this->ZBufferOrigin[1] = static_cast<int>(
static_cast<float>(this->ImageOrigin[1]) * this->ImageSampleDistance);
// Capture the z buffer
this->ZBuffer = ren->GetRenderWindow()->GetZbufferData(x1,y1,x2,y2);
}
this->RayCastFunction->Initialize( ren, vol );
this->RealRayIntegrator->Initialize(vol, this->Scalars);
// Save the volume and mapper temporarily so that they can be accessed later
this->CurrentVolume = vol;
this->CurrentRenderer = ren;
// Create iterators here to prevent race conditions.
this->RayCastIterators
= new vtkUnstructuredGridVolumeRayCastIterator*[this->NumberOfThreads];
for (i = 0; i < this->NumberOfThreads; i++)
{
this->RayCastIterators[i] = this->RayCastFunction->NewIterator();
}
// Set the number of threads to use for ray casting,
// then set the execution method and do it.
this->Threader->SetNumberOfThreads( this->NumberOfThreads );
this->Threader->SetSingleMethod( UnstructuredGridVolumeRayCastMapper_CastRays,
(void *)this);
this->Threader->SingleMethodExecute();
// We don't need these anymore
this->CurrentVolume = NULL;
this->CurrentRenderer = NULL;
for (i = 0; i < this->NumberOfThreads; i++)
{
this->RayCastIterators[i]->Delete();
}
delete[] this->RayCastIterators;
if ( !ren->GetRenderWindow()->GetAbortRender() )
{
float depth;
if ( this->IntermixIntersectingGeometry )
{
depth = this->GetMinimumBoundsDepth( ren, vol );
}
else
{
depth = -1;
}
this->ImageDisplayHelper->
RenderTexture( vol, ren,
this->ImageMemorySize,
this->ImageViewportSize,
this->ImageInUseSize,
this->ImageOrigin,
depth,
this->Image );
this->Timer->StopTimer();
this->TimeToDraw = this->Timer->GetElapsedTime();
this->StoreRenderTime( ren, vol, this->TimeToDraw );
}
else
{
this->ImageSampleDistance = oldImageSampleDistance;
}
if ( this->ZBuffer )
{
delete [] this->ZBuffer;
this->ZBuffer = NULL;
}
this->UpdateProgress(1.0);
}
VTK_THREAD_RETURN_TYPE UnstructuredGridVolumeRayCastMapper_CastRays( void *arg )
{
// Get the info out of the input structure
int threadID = ((vtkMultiThreader::ThreadInfo *)(arg))->ThreadID;
int threadCount = ((vtkMultiThreader::ThreadInfo *)(arg))->NumberOfThreads;
vtkUnstructuredGridVolumeRayCastMapper *me =
(vtkUnstructuredGridVolumeRayCastMapper *)((vtkMultiThreader::ThreadInfo *)arg)->UserData;
if ( !me )
{
vtkGenericWarningMacro("The volume does not have a ray cast mapper!");
return VTK_THREAD_RETURN_VALUE;
}
me->CastRays( threadID, threadCount );
return VTK_THREAD_RETURN_VALUE;
}
template<class T>
inline void vtkUGVRCMLookupCopy(const T *src, T *dest, vtkIdType *lookup,
int numcomponents, int numtuples)
{
for (vtkIdType i = 0; i < numtuples; i++)
{
const T *srctuple = src + lookup[i] * numcomponents;
for (int j = 0; j < numcomponents; j++)
{
*dest = *srctuple;
dest++; srctuple++;
}
}
}
void vtkUnstructuredGridVolumeRayCastMapper::CastRays( int threadID, int threadCount )
{
int i, j;
unsigned char *ucptr;
vtkRenderWindow *renWin = this->CurrentRenderer->GetRenderWindow();
vtkUnstructuredGridVolumeRayCastIterator *iterator
= this->RayCastIterators[threadID];
vtkDoubleArray *intersectionLengths = vtkDoubleArray::New();
vtkDataArray *nearIntersections
= vtkDataArray::CreateDataArray(this->Scalars->GetDataType());
vtkDataArray *farIntersections;
vtkIdList *intersectedCells;
if (this->CellScalars)
{
intersectedCells = vtkIdList::New();
farIntersections = nearIntersections;
}
else
{
farIntersections
= vtkDataArray::CreateDataArray(this->Scalars->GetDataType());
intersectedCells = NULL;
}
for ( j = 0; j < this->ImageInUseSize[1]; j++ )
{
if ( j%threadCount != threadID )
{
continue;
}
if ( !threadID )
{
this->UpdateProgress((double)j/this->ImageInUseSize[1]);
if ( renWin->CheckAbortStatus() )
{
break;
}
}
else if ( renWin->GetAbortRender() )
{
break;
}
ucptr = this->Image + 4*j*this->ImageMemorySize[0];
for ( i = 0; i < this->ImageInUseSize[0]; i++ )
{
int x = i + this->ImageOrigin[0];
int y = j + this->ImageOrigin[1];
double bounds[2] = {0.0,1.0};
float color[4] = {0.0f, 0.0f, 0.0f, 0.0f};
if ( this->ZBuffer )
{
bounds[1] = this->GetZBufferValue( x, y );
}
iterator->SetBounds(bounds);
iterator->Initialize(x, y);
vtkIdType numIntersections;
do
{
if (this->CellScalars)
{
numIntersections = iterator->GetNextIntersections(intersectedCells,
intersectionLengths,
NULL,
NULL, NULL);
nearIntersections
->SetNumberOfComponents(this->Scalars->GetNumberOfComponents());
nearIntersections->SetNumberOfTuples(numIntersections);
switch (this->Scalars->GetDataType())
{
vtkTemplateMacro(vtkUGVRCMLookupCopy
((const VTK_TT*)this->Scalars->GetVoidPointer(0),
(VTK_TT*)nearIntersections->GetVoidPointer(0),
intersectedCells->GetPointer(0),
this->Scalars->GetNumberOfComponents(),
numIntersections));
}
}
else
{
numIntersections = iterator->GetNextIntersections(NULL,
intersectionLengths,
this->Scalars,
nearIntersections,
farIntersections);
}
if (numIntersections < 1) break;
this->RealRayIntegrator->Integrate(intersectionLengths,
nearIntersections,
farIntersections,
color);
} while (color[3] < 0.99);
if ( color[3] > 0.0 )
{
int val;
val = static_cast<int>((color[0]/color[3])*255.0);
val = (val > 255)?(255):(val);
val = (val < 0)?( 0):(val);
ucptr[0] = static_cast<unsigned char>(val);
val = static_cast<int>((color[1]/color[3])*255.0);
val = (val > 255)?(255):(val);
val = (val < 0)?( 0):(val);
ucptr[1] = static_cast<unsigned char>(val);
val = static_cast<int>((color[2]/color[3])*255.0);
val = (val > 255)?(255):(val);
val = (val < 0)?( 0):(val);
ucptr[2] = static_cast<unsigned char>(val);
val = static_cast<int>(color[3]*255.0);
val = (val > 255)?(255):(val);
val = (val < 0)?( 0):(val);
ucptr[3] = static_cast<unsigned char>(val);
}
else
{
ucptr[0] = 0;
ucptr[1] = 0;
ucptr[2] = 0;
ucptr[3] = 0;
}
ucptr+=4;
}
}
intersectionLengths->Delete();
nearIntersections->Delete();
if (!this->CellScalars)
{
farIntersections->Delete();
}
else
{
intersectedCells->Delete();
}
}
double vtkUnstructuredGridVolumeRayCastMapper::
GetMinimumBoundsDepth( vtkRenderer *ren, vtkVolume *vol )
{
double bounds[6];
vol->GetBounds( bounds );
vtkTransform *perspectiveTransform = vtkTransform::New();
vtkMatrix4x4 *perspectiveMatrix = vtkMatrix4x4::New();
ren->ComputeAspect();
double *aspect = ren->GetAspect();
// Get the view matrix in two steps - there is a one step method in camera
// but it turns off stereo so we do not want to use that one
vtkCamera *cam = ren->GetActiveCamera();
perspectiveTransform->Identity();
perspectiveTransform->Concatenate(
cam->GetPerspectiveTransformMatrix(aspect[0]/aspect[1], 0.0, 1.0 ));
perspectiveTransform->Concatenate(cam->GetViewTransformMatrix());
perspectiveMatrix->DeepCopy(perspectiveTransform->GetMatrix());
double minZ = 1.0;
for ( int k = 0; k < 2; k++ )
{
for ( int j = 0; j < 2; j++ )
{
for ( int i = 0; i < 2; i++ )
{
double inPoint[4];
inPoint[0] = bounds[ i];
inPoint[1] = bounds[2+j];
inPoint[2] = bounds[4+k];
inPoint[3] = 1.0;
double outPoint[4];
perspectiveMatrix->MultiplyPoint( inPoint, outPoint );
double testZ = outPoint[2] / outPoint[3];
minZ = ( testZ < minZ ) ? (testZ) : (minZ);
}
}
}
perspectiveTransform->Delete();
perspectiveMatrix->Delete();
return minZ;
}
double vtkUnstructuredGridVolumeRayCastMapper::GetZBufferValue(int x, int y)
{
int xPos, yPos;
xPos = static_cast<int>(static_cast<float>(x) * this->ImageSampleDistance);
yPos = static_cast<int>(static_cast<float>(y) * this->ImageSampleDistance);
xPos = (xPos >= this->ZBufferSize[0])?(this->ZBufferSize[0]-1):(xPos);
yPos = (yPos >= this->ZBufferSize[1])?(this->ZBufferSize[1]-1):(yPos);
return *(this->ZBuffer + yPos*this->ZBufferSize[0] + xPos);
}
// Print method for vtkUnstructuredGridVolumeRayCastMapper
void vtkUnstructuredGridVolumeRayCastMapper::PrintSelf(ostream& os, vtkIndent indent)
{
this->Superclass::PrintSelf(os,indent);
os << indent << "ScalarMode: " << this->GetScalarModeAsString() << endl;
if (this->ArrayAccessMode == VTK_GET_ARRAY_BY_ID)
{
os << indent << "ArrayId: " << this->ArrayId << endl;
}
else
{
os << indent << "ArrayName: " << this->ArrayName << endl;
}
os << indent << "Image Sample Distance: "
<< this->ImageSampleDistance << "\n";
os << indent << "Minimum Image Sample Distance: "
<< this->MinimumImageSampleDistance << "\n";
os << indent << "Maximum Image Sample Distance: "
<< this->MaximumImageSampleDistance << "\n";
os << indent << "Auto Adjust Sample Distances: "
<< this->AutoAdjustSampleDistances << "\n";
os << indent << "Intermix Intersecting Geometry: "
<< (this->IntermixIntersectingGeometry ? "On\n" : "Off\n");
os << indent << "Number Of Threads: " << this->NumberOfThreads << "\n";
if (this->RayCastFunction)
{
os << indent << "RayCastFunction: " <<
this->RayCastFunction->GetClassName() << "\n";
}
else
{
os << indent << "RayCastFunction: (none)\n";
}
if (this->RayIntegrator)
{
os << indent << "RayIntegrator: "
<< this->RayIntegrator->GetClassName() << endl;
}
else
{
os << indent << "RayIntegrator: (automatic)" << endl;
}
// Do not want to print this->ImageOrigin, this->ImageViewportSize or
// this->ImageInUseSize since these are just internal variables with Get
// methods for access from the ray cast function (not part of the public
// API)
}