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.
389 lines
9.7 KiB
389 lines
9.7 KiB
2 years ago
|
/*=========================================================================
|
||
|
|
||
|
Program: Visualization Toolkit
|
||
|
Module: $RCSfile: vtkPNGReader.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 "vtkPNGReader.h"
|
||
|
|
||
|
#include "vtkImageData.h"
|
||
|
#include "vtkObjectFactory.h"
|
||
|
#include "vtkPointData.h"
|
||
|
#include "vtk_png.h"
|
||
|
|
||
|
vtkCxxRevisionMacro(vtkPNGReader, "$Revision: 1.25 $");
|
||
|
vtkStandardNewMacro(vtkPNGReader);
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
// Let us get rid of this funny warning on /W4:
|
||
|
// warning C4611: interaction between '_setjmp' and C++ object
|
||
|
// destruction is non-portable
|
||
|
#pragma warning( disable : 4611 )
|
||
|
#endif
|
||
|
|
||
|
void vtkPNGReader::ExecuteInformation()
|
||
|
{
|
||
|
this->ComputeInternalFileName(this->DataExtent[4]);
|
||
|
if (this->InternalFileName == NULL)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
FILE *fp = fopen(this->InternalFileName, "rb");
|
||
|
if (!fp)
|
||
|
{
|
||
|
vtkErrorMacro("Unable to open file " << this->InternalFileName);
|
||
|
return;
|
||
|
}
|
||
|
unsigned char header[8];
|
||
|
fread(header, 1, 8, fp);
|
||
|
int is_png = !png_sig_cmp(header, 0, 8);
|
||
|
if (!is_png)
|
||
|
{
|
||
|
vtkErrorMacro(<<"Unknown file type! Not a PNG file!");
|
||
|
fclose(fp);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
png_structp png_ptr = png_create_read_struct
|
||
|
(PNG_LIBPNG_VER_STRING, (png_voidp)NULL,
|
||
|
NULL, NULL);
|
||
|
if (!png_ptr)
|
||
|
{
|
||
|
vtkErrorMacro(<< "Out of memory." );
|
||
|
fclose(fp);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
png_infop info_ptr = png_create_info_struct(png_ptr);
|
||
|
if (!info_ptr)
|
||
|
{
|
||
|
png_destroy_read_struct(&png_ptr,
|
||
|
(png_infopp)NULL, (png_infopp)NULL);
|
||
|
vtkErrorMacro(<< "Out of memory.");
|
||
|
fclose(fp);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
png_infop end_info = png_create_info_struct(png_ptr);
|
||
|
if (!end_info)
|
||
|
{
|
||
|
png_destroy_read_struct(&png_ptr, &info_ptr,
|
||
|
(png_infopp)NULL);
|
||
|
vtkErrorMacro(<<"Unable to read PNG file!");
|
||
|
fclose(fp);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Set error handling
|
||
|
if (setjmp (png_jmpbuf(png_ptr)))
|
||
|
{
|
||
|
png_destroy_read_struct (&png_ptr, &info_ptr, (png_infopp)NULL);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
png_init_io(png_ptr, fp);
|
||
|
png_set_sig_bytes(png_ptr, 8);
|
||
|
|
||
|
png_read_info(png_ptr, info_ptr);
|
||
|
|
||
|
png_uint_32 width, height;
|
||
|
int bit_depth, color_type, interlace_type;
|
||
|
int compression_type, filter_method;
|
||
|
// get size and bit-depth of the PNG-image
|
||
|
png_get_IHDR(png_ptr, info_ptr,
|
||
|
&width, &height,
|
||
|
&bit_depth, &color_type, &interlace_type,
|
||
|
&compression_type, &filter_method);
|
||
|
|
||
|
// set-up the transformations
|
||
|
// convert palettes to RGB
|
||
|
if (color_type == PNG_COLOR_TYPE_PALETTE)
|
||
|
{
|
||
|
png_set_palette_to_rgb(png_ptr);
|
||
|
}
|
||
|
|
||
|
// minimum of a byte per pixel
|
||
|
if (color_type == PNG_COLOR_TYPE_GRAY && bit_depth < 8)
|
||
|
{
|
||
|
png_set_gray_1_2_4_to_8(png_ptr);
|
||
|
}
|
||
|
|
||
|
// add alpha if any alpha found
|
||
|
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS))
|
||
|
{
|
||
|
png_set_tRNS_to_alpha(png_ptr);
|
||
|
}
|
||
|
|
||
|
// update the info now that we have defined the filters
|
||
|
png_read_update_info(png_ptr, info_ptr);
|
||
|
|
||
|
this->DataExtent[0] = 0;
|
||
|
this->DataExtent[1] = width - 1;
|
||
|
this->DataExtent[2] = 0;
|
||
|
this->DataExtent[3] = height - 1;
|
||
|
|
||
|
if (bit_depth <= 8)
|
||
|
{
|
||
|
this->SetDataScalarTypeToUnsignedChar();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
this->SetDataScalarTypeToUnsignedShort();
|
||
|
}
|
||
|
this->SetNumberOfScalarComponents(
|
||
|
png_get_channels(png_ptr, info_ptr));
|
||
|
this->vtkImageReader2::ExecuteInformation();
|
||
|
|
||
|
// close the file
|
||
|
png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
|
||
|
fclose(fp);
|
||
|
}
|
||
|
|
||
|
|
||
|
template <class OT>
|
||
|
void vtkPNGReaderUpdate2(vtkPNGReader *self, OT *outPtr,
|
||
|
int *outExt, vtkIdType *outInc, long pixSize)
|
||
|
{
|
||
|
unsigned int ui;
|
||
|
int i;
|
||
|
FILE *fp = fopen(self->GetInternalFileName(), "rb");
|
||
|
if (!fp)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
unsigned char header[8];
|
||
|
fread(header, 1, 8, fp);
|
||
|
int is_png = !png_sig_cmp(header, 0, 8);
|
||
|
if (!is_png)
|
||
|
{
|
||
|
fclose(fp);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
png_structp png_ptr = png_create_read_struct
|
||
|
(PNG_LIBPNG_VER_STRING, (png_voidp)NULL, NULL, NULL);
|
||
|
if (!png_ptr)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
png_infop info_ptr = png_create_info_struct(png_ptr);
|
||
|
if (!info_ptr)
|
||
|
{
|
||
|
png_destroy_read_struct(&png_ptr,
|
||
|
(png_infopp)NULL, (png_infopp)NULL);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
png_infop end_info = png_create_info_struct(png_ptr);
|
||
|
if (!end_info)
|
||
|
{
|
||
|
png_destroy_read_struct(&png_ptr, &info_ptr,
|
||
|
(png_infopp)NULL);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Set error handling
|
||
|
if (setjmp (png_jmpbuf(png_ptr)))
|
||
|
{
|
||
|
png_destroy_read_struct (&png_ptr, &info_ptr, (png_infopp)NULL);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
png_init_io(png_ptr, fp);
|
||
|
png_set_sig_bytes(png_ptr, 8);
|
||
|
png_read_info(png_ptr, info_ptr);
|
||
|
|
||
|
png_uint_32 width, height;
|
||
|
int bit_depth, color_type, interlace_type;
|
||
|
int compression_type, filter_method;
|
||
|
// get size and bit-depth of the PNG-image
|
||
|
png_get_IHDR(png_ptr, info_ptr,
|
||
|
&width, &height,
|
||
|
&bit_depth, &color_type, &interlace_type,
|
||
|
&compression_type, &filter_method);
|
||
|
|
||
|
// set-up the transformations
|
||
|
// convert palettes to RGB
|
||
|
if (color_type == PNG_COLOR_TYPE_PALETTE)
|
||
|
{
|
||
|
png_set_palette_to_rgb(png_ptr);
|
||
|
}
|
||
|
|
||
|
// minimum of a byte per pixel
|
||
|
if (color_type == PNG_COLOR_TYPE_GRAY && bit_depth < 8)
|
||
|
{
|
||
|
png_set_gray_1_2_4_to_8(png_ptr);
|
||
|
}
|
||
|
|
||
|
// add alpha if any alpha found
|
||
|
if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS))
|
||
|
{
|
||
|
png_set_tRNS_to_alpha(png_ptr);
|
||
|
}
|
||
|
|
||
|
if (bit_depth > 8)
|
||
|
{
|
||
|
#ifndef VTK_WORDS_BIGENDIAN
|
||
|
png_set_swap(png_ptr);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
// have libpng handle interlacing
|
||
|
//int number_of_passes = png_set_interlace_handling(png_ptr);
|
||
|
// update the info now that we have defined the filters
|
||
|
png_read_update_info(png_ptr, info_ptr);
|
||
|
|
||
|
int rowbytes = png_get_rowbytes(png_ptr, info_ptr);
|
||
|
unsigned char *tempImage = new unsigned char [rowbytes*height];
|
||
|
png_bytep *row_pointers = new png_bytep [height];
|
||
|
for (ui = 0; ui < height; ++ui)
|
||
|
{
|
||
|
row_pointers[ui] = tempImage + rowbytes*ui;
|
||
|
}
|
||
|
png_read_image(png_ptr, row_pointers);
|
||
|
|
||
|
// copy the data into the outPtr
|
||
|
OT *outPtr2;
|
||
|
outPtr2 = outPtr;
|
||
|
long outSize = pixSize*(outExt[1] - outExt[0] + 1);
|
||
|
for (i = outExt[2]; i <= outExt[3]; ++i)
|
||
|
{
|
||
|
memcpy(outPtr2,row_pointers[height - i - 1] + outExt[0]*pixSize,outSize);
|
||
|
outPtr2 += outInc[1];
|
||
|
}
|
||
|
delete [] tempImage;
|
||
|
delete [] row_pointers;
|
||
|
|
||
|
// close the file
|
||
|
png_read_end(png_ptr, NULL);
|
||
|
png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
|
||
|
fclose(fp);
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
// This function reads in one data of data.
|
||
|
// templated to handle different data types.
|
||
|
template <class OT>
|
||
|
void vtkPNGReaderUpdate(vtkPNGReader *self, vtkImageData *data, OT *outPtr)
|
||
|
{
|
||
|
vtkIdType outIncr[3];
|
||
|
int outExtent[6];
|
||
|
OT *outPtr2;
|
||
|
|
||
|
data->GetExtent(outExtent);
|
||
|
data->GetIncrements(outIncr);
|
||
|
|
||
|
long pixSize = data->GetNumberOfScalarComponents()*sizeof(OT);
|
||
|
|
||
|
outPtr2 = outPtr;
|
||
|
int idx2;
|
||
|
for (idx2 = outExtent[4]; idx2 <= outExtent[5]; ++idx2)
|
||
|
{
|
||
|
self->ComputeInternalFileName(idx2);
|
||
|
// read in a PNG file
|
||
|
vtkPNGReaderUpdate2(self, outPtr2, outExtent, outIncr, pixSize);
|
||
|
self->UpdateProgress((idx2 - outExtent[4])/
|
||
|
(outExtent[5] - outExtent[4] + 1.0));
|
||
|
outPtr2 += outIncr[2];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
// This function reads a data from a file. The datas extent/axes
|
||
|
// are assumed to be the same as the file extent/order.
|
||
|
void vtkPNGReader::ExecuteData(vtkDataObject *output)
|
||
|
{
|
||
|
vtkImageData *data = this->AllocateOutputData(output);
|
||
|
|
||
|
if (this->InternalFileName == NULL)
|
||
|
{
|
||
|
vtkErrorMacro(<< "Either a FileName or FilePrefix must be specified.");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
data->GetPointData()->GetScalars()->SetName("PNGImage");
|
||
|
|
||
|
this->ComputeDataIncrements();
|
||
|
|
||
|
// Call the correct templated function for the output
|
||
|
void *outPtr;
|
||
|
|
||
|
// Call the correct templated function for the input
|
||
|
outPtr = data->GetScalarPointer();
|
||
|
switch (data->GetScalarType())
|
||
|
{
|
||
|
vtkTemplateMacro(vtkPNGReaderUpdate(this, data, (VTK_TT *)(outPtr)));
|
||
|
default:
|
||
|
vtkErrorMacro(<< "UpdateFromFile: Unknown data type");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
int vtkPNGReader::CanReadFile(const char* fname)
|
||
|
{
|
||
|
FILE* fp = fopen(fname, "rb");
|
||
|
if(!fp)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
unsigned char header[8];
|
||
|
fread(header, 1, 8, fp);
|
||
|
int is_png = !png_sig_cmp(header, 0, 8);
|
||
|
if(!is_png)
|
||
|
{
|
||
|
fclose(fp);
|
||
|
return 0;
|
||
|
}
|
||
|
png_structp png_ptr = png_create_read_struct
|
||
|
(PNG_LIBPNG_VER_STRING, (png_voidp)NULL,
|
||
|
NULL, NULL);
|
||
|
if (!png_ptr)
|
||
|
{
|
||
|
fclose(fp);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
png_infop info_ptr = png_create_info_struct(png_ptr);
|
||
|
if (!info_ptr)
|
||
|
{
|
||
|
png_destroy_read_struct(&png_ptr,
|
||
|
(png_infopp)NULL, (png_infopp)NULL);
|
||
|
fclose(fp);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
png_infop end_info = png_create_info_struct(png_ptr);
|
||
|
if (!end_info)
|
||
|
{
|
||
|
png_destroy_read_struct(&png_ptr, &info_ptr,
|
||
|
(png_infopp)NULL);
|
||
|
fclose(fp);
|
||
|
return 0;
|
||
|
}
|
||
|
png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
|
||
|
|
||
|
fclose(fp);
|
||
|
return 3;
|
||
|
}
|
||
|
#ifdef _MSC_VER
|
||
|
// Put the warning back
|
||
|
#pragma warning( default : 4611 )
|
||
|
#endif
|
||
|
|
||
|
//----------------------------------------------------------------------------
|
||
|
void vtkPNGReader::PrintSelf(ostream& os, vtkIndent indent)
|
||
|
{
|
||
|
this->Superclass::PrintSelf(os,indent);
|
||
|
}
|