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.
 
 
 
 
 
 

472 lines
14 KiB

/*=========================================================================
Program: Visualization Toolkit
Module: $RCSfile: vtkImageImport.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 "vtkImageImport.h"
#include "vtkByteSwap.h"
#include "vtkImageData.h"
#include "vtkImageImportExecutive.h"
#include "vtkInformation.h"
#include "vtkInformationVector.h"
#include "vtkObjectFactory.h"
#include "vtkStreamingDemandDrivenPipeline.h"
#include "vtkPointData.h"
#include <ctype.h>
#include <vtkstd/exception>
vtkCxxRevisionMacro(vtkImageImport, "$Revision: 1.48 $");
vtkStandardNewMacro(vtkImageImport);
#define tryCatchMacro(invocation, messagePrepend)\
try\
{\
invocation;\
}\
catch (vtkstd::exception &_e)\
{\
vtkErrorMacro(<<messagePrepend <<_e.what());\
}\
catch (...)\
{\
vtkErrorMacro(<<"Unknown exception.");\
}\
//----------------------------------------------------------------------------
vtkImageImport::vtkImageImport()
{
int idx;
this->ImportVoidPointer = 0;
this->DataScalarType = VTK_SHORT;
this->NumberOfScalarComponents = 1;
for (idx = 0; idx < 3; ++idx)
{
this->DataExtent[idx*2] = this->DataExtent[idx*2 + 1] = 0;
this->WholeExtent[idx*2] = this->WholeExtent[idx*2 + 1] = 0;
this->DataSpacing[idx] = 1.0;
this->DataOrigin[idx] = 0.0;
}
this->SaveUserArray = 0;
this->CallbackUserData = 0;
this->UpdateInformationCallback = 0;
this->PipelineModifiedCallback = 0;
this->WholeExtentCallback = 0;
this->SpacingCallback = 0;
this->OriginCallback = 0;
this->ScalarTypeCallback = 0;
this->NumberOfComponentsCallback = 0;
this->PropagateUpdateExtentCallback = 0;
this->UpdateDataCallback = 0;
this->DataExtentCallback = 0;
this->BufferPointerCallback = 0;
this->SetNumberOfInputPorts(0);
vtkExecutive *exec = vtkImageImportExecutive::New();
this->SetExecutive(exec);
exec->Delete();
}
//----------------------------------------------------------------------------
vtkImageImport::~vtkImageImport()
{
if ((this->ImportVoidPointer) && (!this->SaveUserArray))
{
delete [] (char *)this->ImportVoidPointer;
}
}
//----------------------------------------------------------------------------
void vtkImageImport::PrintSelf(ostream& os, vtkIndent indent)
{
this->Superclass::PrintSelf(os,indent);
int idx;
os << indent << "ImportVoidPointer: " << this->ImportVoidPointer << "\n";
os << indent << "DataScalarType: "
<< vtkImageScalarTypeNameMacro(this->DataScalarType) << "\n";
os << indent << "NumberOfScalarComponents: "
<< this->NumberOfScalarComponents << "\n";
os << indent << "WholeExtent: (" << this->WholeExtent[0];
for (idx = 1; idx < 6; ++idx)
{
os << ", " << this->WholeExtent[idx];
}
os << ")\n";
os << indent << "DataExtent: (" << this->DataExtent[0];
for (idx = 1; idx < 6; ++idx)
{
os << ", " << this->DataExtent[idx];
}
os << ")\n";
os << indent << "DataSpacing: (" << this->DataSpacing[0];
for (idx = 1; idx < 3; ++idx)
{
os << ", " << this->DataSpacing[idx];
}
os << ")\n";
os << indent << "DataOrigin: (" << this->DataOrigin[0];
for (idx = 1; idx < 3; ++idx)
{
os << ", " << this->DataOrigin[idx];
}
os << ")\n";
os << indent << "CallbackUserData: "
<< (this->CallbackUserData? "Set" : "Not Set") << "\n";
os << indent << "UpdateInformationCallback: "
<< (this->UpdateInformationCallback? "Set" : "Not Set") << "\n";
os << indent << "PipelineModifiedCallback: "
<< (this->PipelineModifiedCallback? "Set" : "Not Set") << "\n";
os << indent << "WholeExtentCallback: "
<< (this->WholeExtentCallback? "Set" : "Not Set") << "\n";
os << indent << "SpacingCallback: "
<< (this->SpacingCallback? "Set" : "Not Set") << "\n";
os << indent << "OriginCallback: "
<< (this->OriginCallback? "Set" : "Not Set") << "\n";
os << indent << "ScalarTypeCallback: "
<< (this->ScalarTypeCallback? "Set" : "Not Set") << "\n";
os << indent << "NumberOfComponentsCallback: "
<< (this->NumberOfComponentsCallback? "Set" : "Not Set") << "\n";
os << indent << "PropagateUpdateExtentCallback: "
<< (this->PropagateUpdateExtentCallback? "Set" : "Not Set") << "\n";
os << indent << "UpdateDataCallback: "
<< (this->UpdateDataCallback? "Set" : "Not Set") << "\n";
os << indent << "DataExtentCallback: "
<< (this->DataExtentCallback? "Set" : "Not Set") << "\n";
os << indent << "BufferPointerCallback: "
<< (this->BufferPointerCallback? "Set" : "Not Set") << "\n";
}
//----------------------------------------------------------------------------
int vtkImageImport::RequestUpdateExtent(
vtkInformation* vtkNotUsed(request),
vtkInformationVector** vtkNotUsed(inputVector),
vtkInformationVector* outputVector)
{
if (this->PropagateUpdateExtentCallback)
{
int uExt[6];
vtkInformation* outInfo = outputVector->GetInformationObject(0);
outInfo->Get(vtkStreamingDemandDrivenPipeline::UPDATE_EXTENT(),uExt);
tryCatchMacro(
(this->PropagateUpdateExtentCallback)(this->CallbackUserData,uExt),
"PropagateUpdateExtentCallback: ");
}
return 1;
}
//----------------------------------------------------------------------------
int vtkImageImport::RequestInformation (
vtkInformation * vtkNotUsed(request),
vtkInformationVector ** vtkNotUsed( inputVector ),
vtkInformationVector *outputVector)
{
// get the info objects
vtkInformation* outInfo = outputVector->GetInformationObject(0);
// If set, use the callbacks to fill in our data members.
this->InvokeExecuteInformationCallbacks();
// Legacy support for code that sets only DataExtent.
this->LegacyCheckWholeExtent();
// set the whole extent
outInfo->Set(vtkStreamingDemandDrivenPipeline::WHOLE_EXTENT(),
this->WholeExtent,6);
// set the spacing
outInfo->Set(vtkDataObject::SPACING(),this->DataSpacing,3);
// set the origin.
outInfo->Set(vtkDataObject::ORIGIN(),this->DataOrigin,3);
// set data type
vtkDataObject::SetPointDataActiveScalarInfo(outInfo, this->DataScalarType,
this->NumberOfScalarComponents);
return 1;
}
//----------------------------------------------------------------------------
void vtkImageImport::ExecuteData(vtkDataObject *output)
{
// If set, use the callbacks to prepare our input data.
this->InvokeExecuteDataCallbacks();
vtkImageData *data = vtkImageData::SafeDownCast(output);
data->SetExtent(0,0,0,0,0,0);
data->AllocateScalars();
void *ptr = this->GetImportVoidPointer();
int size =
(this->DataExtent[1] - this->DataExtent[0]+1) *
(this->DataExtent[3] - this->DataExtent[2]+1) *
(this->DataExtent[5] - this->DataExtent[4]+1) *
this->NumberOfScalarComponents;
data->SetExtent(this->DataExtent);
data->GetPointData()->GetScalars()->SetVoidArray(ptr,size,1);
}
//----------------------------------------------------------------------------
void vtkImageImport::CopyImportVoidPointer(void *ptr, int size)
{
unsigned char *mem = new unsigned char [size];
memcpy(mem,ptr,size);
this->SetImportVoidPointer(mem,0);
}
//----------------------------------------------------------------------------
void vtkImageImport::SetImportVoidPointer(void *ptr)
{
this->SetImportVoidPointer(ptr,1);
}
//----------------------------------------------------------------------------
void vtkImageImport::SetImportVoidPointer(void *ptr, int save)
{
if (ptr != this->ImportVoidPointer)
{
if ((this->ImportVoidPointer) && (!this->SaveUserArray))
{
vtkDebugMacro (<< "Deleting the array...");
delete [] (char *)this->ImportVoidPointer;
}
else
{
vtkDebugMacro (<<"Warning, array not deleted, but will point to new array.");
}
this->Modified();
}
this->SaveUserArray = save;
this->ImportVoidPointer = ptr;
}
//----------------------------------------------------------------------------
int vtkImageImport::InvokePipelineModifiedCallbacks()
{
if (this->PipelineModifiedCallback)
{
int ret;
try
{
ret = (this->PipelineModifiedCallback)(this->CallbackUserData);
}
catch (vtkstd::exception &_e)
{
vtkErrorMacro(<<"Calling PipelineModifiedCallback: " << _e.what());
// if an error occurred, we don't want the pipeline to run again
// until the error has been rectified. It can be assumed that
// the rectifying actions will set the modified flag.
ret = 0;
}
catch (...)
{
vtkErrorMacro(<<"Unknown exception.");
// same logic as above
ret = 0;
}
return ret;
}
else
{
// if there's no PipelineModified installed, we return 0
return 0;
}
}
//----------------------------------------------------------------------------
void vtkImageImport::InvokeUpdateInformationCallbacks()
{
if (this->UpdateInformationCallback)
{
tryCatchMacro((this->UpdateInformationCallback)(this->CallbackUserData),
"Calling UpdateInformationCallback: ");
}
if (this->InvokePipelineModifiedCallbacks())
{
this->Modified();
}
}
//----------------------------------------------------------------------------
void vtkImageImport::InvokeExecuteInformationCallbacks()
{
if (this->WholeExtentCallback)
{
tryCatchMacro(
this->SetWholeExtent(
(this->WholeExtentCallback)(this->CallbackUserData)),
"Calling WholeExtentCallback: ");
}
if (this->SpacingCallback)
{
tryCatchMacro(
this->SetDataSpacing((this->SpacingCallback)(this->CallbackUserData)),
"Calling SpacingCallback: ");
}
if (this->OriginCallback)
{
tryCatchMacro(
this->SetDataOrigin((this->OriginCallback)(this->CallbackUserData)),
"Calling OriginCallback: ");
}
if (this->NumberOfComponentsCallback)
{
tryCatchMacro(
this->SetNumberOfScalarComponents(
(this->NumberOfComponentsCallback)(this->CallbackUserData)),
"Calling NumberOfComponentsCallback: ");
}
if (this->ScalarTypeCallback)
{
const char* scalarType = "double"; // default
tryCatchMacro(
scalarType = (this->ScalarTypeCallback)(this->CallbackUserData),
"Calling ScalarTypeCallback: ");
if (strcmp(scalarType, "double")==0)
{
this->SetDataScalarType(VTK_DOUBLE);
}
else if (strcmp(scalarType, "float")==0)
{
this->SetDataScalarType(VTK_FLOAT);
}
else if (strcmp(scalarType, "long")==0)
{
this->SetDataScalarType(VTK_LONG);
}
else if (strcmp(scalarType, "unsigned long")==0)
{
this->SetDataScalarType(VTK_UNSIGNED_LONG);
}
else if (strcmp(scalarType, "int")==0)
{
this->SetDataScalarType(VTK_INT);
}
else if (strcmp(scalarType, "unsigned int")==0)
{
this->SetDataScalarType(VTK_UNSIGNED_INT);
}
else if (strcmp(scalarType, "short")==0)
{
this->SetDataScalarType(VTK_SHORT);
}
else if (strcmp(scalarType, "unsigned short")==0)
{
this->SetDataScalarType(VTK_UNSIGNED_SHORT);
}
else if (strcmp(scalarType, "char")==0)
{
this->SetDataScalarType(VTK_CHAR);
}
else if (strcmp(scalarType, "unsigned char")==0)
{
this->SetDataScalarType(VTK_UNSIGNED_CHAR);
}
}
}
//----------------------------------------------------------------------------
void vtkImageImport::InvokeExecuteDataCallbacks()
{
if (this->UpdateDataCallback)
{
tryCatchMacro(
(this->UpdateDataCallback)(this->CallbackUserData),
"Calling UpdateDataCallback: ");
}
if (this->DataExtentCallback)
{
tryCatchMacro(
this->SetDataExtent((this->DataExtentCallback)(this->CallbackUserData)),
"Calling DataExtentCallback: ");
}
if (this->BufferPointerCallback)
{
tryCatchMacro(
this->SetImportVoidPointer(
(this->BufferPointerCallback)(this->CallbackUserData)),
"Calling BufferPointerCallback: ");
}
}
//----------------------------------------------------------------------------
// In the past, this class made no distinction between whole extent and
// buffered extent, so only SetDataExtent also set the whole extent of
// the output. Now, there is a separate SetWholeExtent which should be
// called as well.
void vtkImageImport::LegacyCheckWholeExtent()
{
// If the WholeExtentCallback is set, this must not be legacy code.
if (this->WholeExtentCallback)
{
return;
}
int i;
// Check whether the whole extent has been set.
for(i=0; i < 6; ++i)
{
if (this->WholeExtent[i] != 0)
{
return;
}
}
// The whole extent has not been set. Copy it from the data extent
// and issue a warning.
for(i=0; i < 6; ++i)
{
this->WholeExtent[i] = this->DataExtent[i];
}
vtkWarningMacro("\n"
"There is a distinction between the whole extent and the buffered\n"
"extent of an imported image. Use SetWholeExtent to set the extent\n"
"of the entire image. Use SetDataExtent to set the extent of the\n"
"portion of the image that is in the buffer set with\n"
"SetImportVoidPointer. Both should be called even if the extents are\n"
"the same.");
}