You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

262 lines
8.8 KiB
C

/*=========================================================================
Program: Visualization Toolkit
Module: vtkNIFTIImageReader.h
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.
=========================================================================*/
/**
* @class vtkNIFTIImageReader
* @brief Read NIfTI-1 and NIfTI-2 medical image files
*
* This class reads NIFTI files, either in .nii format or as separate
* .img and .hdr files. If two files are used, then they can be passed
* by using SetFileNames() instead of SetFileName(). Files ending in .gz
* are decompressed on-the-fly while they are being read. Files with
* complex numbers or vector dimensions will be read as multi-component
* images. If a NIFTI file has a time dimension, then by default only the
* first image in the time series will be read, but the TimeAsVector
* flag can be set to read the time steps as vector components. Files in
* Analyze 7.5 format are also supported by this reader.
* @par Thanks:
* This class was contributed to VTK by the Calgary Image Processing and
* Analysis Centre (CIPAC).
* @sa
* vtkNIFTIImageWriter, vtkNIFTIImageHeader
*/
#ifndef vtkNIFTIImageReader_h
#define vtkNIFTIImageReader_h
#include "vtkIOImageModule.h" // For export macro
#include "vtkImageReader2.h"
class vtkNIFTIImageHeader;
class vtkMatrix4x4;
struct nifti_1_header;
//----------------------------------------------------------------------------
class VTKIOIMAGE_EXPORT vtkNIFTIImageReader : public vtkImageReader2
{
public:
//@{
/**
* Static method for construction.
*/
static vtkNIFTIImageReader* New();
vtkTypeMacro(vtkNIFTIImageReader, vtkImageReader2);
//@}
/**
* Print information about this object.
*/
void PrintSelf(ostream& os, vtkIndent indent) override;
/**
* Valid extensions for this file type.
*/
const char* GetFileExtensions() override { return ".nii .nii.gz .img .img.gz .hdr .hdr.gz"; }
/**
* Return a descriptive name that might be useful in a GUI.
*/
const char* GetDescriptiveName() override { return "NIfTI"; }
/**
* Return true if this reader can read the given file.
*/
int CanReadFile(const char* filename) override;
//@{
/**
* Read the time dimension as scalar components (default: Off).
* If this is on, then each time point will be stored as a component in
* the image data. If the file has both a time dimension and a vector
* dimension, then the number of components will be the product of these
* two dimensions, i.e. the components will store a sequence of vectors.
*/
vtkGetMacro(TimeAsVector, bool);
vtkSetMacro(TimeAsVector, bool);
vtkBooleanMacro(TimeAsVector, bool);
//@}
/**
* Get the time dimension that was stored in the NIFTI header.
*/
int GetTimeDimension() { return this->Dim[4]; }
double GetTimeSpacing() { return this->PixDim[4]; }
/**
* Get the slope and intercept for rescaling the scalar values.
* These values allow calibration of the data to real values.
* Use the equation v = u*RescaleSlope + RescaleIntercept.
* This directly returns the values stored in the scl_slope and
* scl_inter fields in the NIFTI header.
*/
double GetRescaleSlope() { return this->RescaleSlope; }
double GetRescaleIntercept() { return this->RescaleIntercept; }
//@{
/**
* Read planar RGB (separate R, G, and B planes), rather than packed RGB.
* The NIFTI format should always use packed RGB. The Analyze format,
* however, was used to store both planar RGB and packed RGB depending
* on the software, without any indication in the header about which
* convention was being used. Use this if you have a planar RGB file.
*/
vtkGetMacro(PlanarRGB, bool);
vtkSetMacro(PlanarRGB, bool);
vtkBooleanMacro(PlanarRGB, bool);
//@}
/**
* QFac gives the slice order in the NIFTI file versus the VTK image.
* If QFac is -1, then the VTK slice index K is related to the NIFTI
* slice index k by the equation K = (num_slices - k - 1). VTK requires
* the slices to be ordered so that the voxel indices (I,J,K) provide a
* right-handed coordinate system, whereas NIFTI does not. Instead,
* NIFTI stores a factor called "qfac" in the header to signal when the
* (i,j,k) indices form a left-handed coordinate system. QFac will only
* ever have values of +1 or -1.
*/
double GetQFac() { return this->QFac; }
/**
* Get a matrix that gives the "qform" orientation and offset for the data.
* If no qform matrix was stored in the file, the return value is nullptr.
* This matrix will transform VTK data coordinates into the NIFTI oriented
* data coordinates, where +X points right, +Y points anterior (toward the
* front), and +Z points superior (toward the head). The qform matrix will
* always have a positive determinant. The offset that is stored in the
* matrix gives the position of the first pixel in the first slice of the
* VTK image data. Note that if QFac is -1, then the first slice in the
* VTK image data is the last slice in the NIFTI file, and the Z offset
* will automatically be adjusted to compensate for this.
*/
vtkMatrix4x4* GetQFormMatrix() { return this->QFormMatrix; }
/**
* Get a matrix that gives the "sform" orientation and offset for the data.
* If no sform matrix was stored in the file, the return value is nullptr.
* Like the qform matrix, this matrix will transform VTK data coordinates
* into a NIFTI coordinate system. Unlike the qform matrix, the sform
* matrix can contain scaling information and can even (rarely) have
* a negative determinant, i.e. a flip. This matrix is modified slightly
* as compared to the sform matrix stored in the NIFTI header: the pixdim
* pixel spacing is factored out. Also, if QFac is -1, then the VTK slices
* are in reverse order as compared to the NIFTI slices, hence as compared
* to the sform matrix stored in the header, the third column of this matrix
* is multiplied by -1 and the Z offset is shifted to compensate for the
* fact that the last slice has become the first.
*/
vtkMatrix4x4* GetSFormMatrix() { return this->SFormMatrix; }
/**
* Get the raw header information from the NIfTI file.
*/
vtkNIFTIImageHeader* GetNIFTIHeader();
protected:
vtkNIFTIImageReader();
~vtkNIFTIImageReader() override;
/**
* Read the header information.
*/
int RequestInformation(vtkInformation* request, vtkInformationVector** inputVector,
vtkInformationVector* outputVector) override;
/**
* Read the voxel data.
*/
int RequestData(vtkInformation* request, vtkInformationVector** inputVector,
vtkInformationVector* outputVector) override;
/**
* Do a case-insensitive check for the given extension.
* The check will succeed if the filename ends in ".gz", and if the
* extension matches after removing the ".gz".
*/
static bool CheckExtension(const char* fname, const char* ext);
/**
* Make a new filename by replacing extension "ext1" with "ext2".
* The extensions must include a period, must be three characters
* long, and must be lower case. This method also verifies that
* the file exists, and adds or subtracts a ".gz" as necessary
* If the file exists, a new string is returned that must be
* deleted by the caller. Otherwise, the return value is nullptr.
*/
static char* ReplaceExtension(const char* fname, const char* ext1, const char* ext2);
/**
* Check the version of the header.
*/
static int CheckNIFTIVersion(const nifti_1_header* hdr);
/**
* Return true if an Analyze 7.5 header was found.
*/
static bool CheckAnalyzeHeader(const nifti_1_header* hdr);
/**
* Read the time dimension as if it was a vector dimension.
*/
bool TimeAsVector;
//@{
/**
* Information for rescaling data to quantitative units.
*/
double RescaleIntercept;
double RescaleSlope;
//@}
/**
* Is -1 if VTK slice order is opposite to NIFTI slice order, +1 otherwise.
*/
double QFac;
//@{
/**
* The orientation matrices for the NIFTI file.
*/
vtkMatrix4x4* QFormMatrix;
vtkMatrix4x4* SFormMatrix;
//@}
/**
* The dimensions of the NIFTI file.
*/
int Dim[8];
/**
* The spacings in the NIFTI file.
*/
double PixDim[8];
/**
* A copy of the header from the file that was most recently read.
*/
vtkNIFTIImageHeader* NIFTIHeader;
/**
* Use planar RGB instead of the default (packed).
*/
bool PlanarRGB;
private:
vtkNIFTIImageReader(const vtkNIFTIImageReader&) = delete;
void operator=(const vtkNIFTIImageReader&) = delete;
};
#endif // vtkNIFTIImageReader_h