// Copyright (C) 2008 Davis E. King (davis@dlib.net), Nils Labugt
// License: Boost Software License See LICENSE.txt for the full license.
#ifndef DLIB_PNG_IMPORT
#define DLIB_PNG_IMPORT
#include <memory>
#include "png_loader_abstract.h"
#include "image_loader.h"
#include "../pixel.h"
#include "../dir_nav.h"
#include "../test_for_odr_violations.h"
namespace dlib
{
struct LibpngData;
struct PngBufferReaderState;
struct FileInfo;
class png_loader : noncopyable
{
public:
png_loader( const char* filename );
png_loader( const std::string& filename );
png_loader( const dlib::file& f );
png_loader( const unsigned char* image_buffer, size_t buffer_size );
~png_loader();
bool is_gray() const;
bool is_graya() const;
bool is_rgb() const;
bool is_rgba() const;
unsigned int bit_depth () const { return bit_depth_; }
template<typename T>
void get_image( T& t_) const
{
#ifndef DLIB_PNG_SUPPORT
/* !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
You are getting this error because you are trying to use the png_loader
object but you haven't defined DLIB_PNG_SUPPORT. You must do so to use
this object. You must also make sure you set your build environment
to link against the libpng library.
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!*/
COMPILE_TIME_ASSERT(sizeof(T) == 0);
#endif
typedef typename image_traits<T>::pixel_type pixel_type;
image_view<T> t(t_);
t.set_size( height_, width_ );
if (is_gray() && bit_depth_ == 8)
{
for ( unsigned n = 0; n < height_;n++ )
{
const unsigned char* v = get_row( n );
for ( unsigned m = 0; m < width_;m++ )
{
unsigned char p = v[m];
assign_pixel( t[n][m], p );
}
}
}
else if (is_gray() && bit_depth_ == 16)
{
for ( unsigned n = 0; n < height_;n++ )
{
const uint16* v = (uint16*)get_row( n );
for ( unsigned m = 0; m < width_;m++ )
{
dlib::uint16 p = v[m];
assign_pixel( t[n][m], p );
}
}
}
else if (is_graya() && bit_depth_ == 8)
{
for ( unsigned n = 0; n < height_;n++ )
{
const unsigned char* v = get_row( n );
for ( unsigned m = 0; m < width_; m++ )
{
unsigned char p = v[m*2];
if (!pixel_traits<pixel_type>::has_alpha)
{
assign_pixel( t[n][m], p );
}
else
{
unsigned char pa = v[m*2+1];
rgb_alpha_pixel pix;
assign_pixel(pix, p);
assign_pixel(pix.alpha, pa);
assign_pixel(t[n][m], pix);
}
}
}
}
else if (is_graya() && bit_depth_ == 16)
{
for ( unsigned n = 0; n < height_;n++ )
{
const uint16* v = (uint16*)get_row( n );
for ( unsigned m = 0; m < width_; m++ )
{
dlib::uint16 p = v[m*2];
if (!pixel_traits<pixel_type>::has_alpha)
{
assign_pixel( t[n][m], p );
}
else
{
dlib::uint16 pa = v[m*2+1];
rgb_alpha_pixel pix;
assign_pixel(pix, p);
assign_pixel(pix.alpha, pa);
assign_pixel(t[n][m], pix);
}
}
}
}
else if (is_rgb() && bit_depth_ == 8)
{
for ( unsigned n = 0; n < height_;n++ )
{
const unsigned char* v = get_row( n );
for ( unsigned m = 0; m < width_;m++ )
{
rgb_pixel p;
p.red = v[m*3];
p.green = v[m*3+1];
p.blue = v[m*3+2];
assign_pixel( t[n][m], p );
}
}
}
else if (is_rgb() && bit_depth_ == 16)
{
for ( unsigned n = 0; n < height_;n++ )
{
const uint16* v = (uint16*)get_row( n );
for ( unsigned m = 0; m < width_;m++ )
{
rgb_pixel p;
p.red = static_cast<uint8>(v[m*3]);
p.green = static_cast<uint8>(v[m*3+1]);
p.blue = static_cast<uint8>(v[m*3+2]);
assign_pixel( t[n][m], p );
}
}
}
else if (is_rgba() && bit_depth_ == 8)
{
if (!pixel_traits<pixel_type>::has_alpha)
assign_all_pixels(t,0);
for ( unsigned n = 0; n < height_;n++ )
{
const unsigned char* v = get_row( n );
for ( unsigned m = 0; m < width_;m++ )
{
rgb_alpha_pixel p;
p.red = v[m*4];
p.green = v[m*4+1];
p.blue = v[m*4+2];
p.alpha = v[m*4+3];
assign_pixel( t[n][m], p );
}
}
}
else if (is_rgba() && bit_depth_ == 16)
{
if (!pixel_traits<pixel_type>::has_alpha)
assign_all_pixels(t,0);
for ( unsigned n = 0; n < height_;n++ )
{
const uint16* v = (uint16*)get_row( n );
for ( unsigned m = 0; m < width_;m++ )
{
rgb_alpha_pixel p;
p.red = static_cast<uint8>(v[m*4]);
p.green = static_cast<uint8>(v[m*4+1]);
p.blue = static_cast<uint8>(v[m*4+2]);
p.alpha = static_cast<uint8>(v[m*4+3]);
assign_pixel( t[n][m], p );
}
}
}
}
private:
const unsigned char* get_row( unsigned i ) const;
std::unique_ptr<FileInfo> check_file( const char* filename );
void read_image( std::unique_ptr<FileInfo> file_info );
unsigned height_, width_;
unsigned bit_depth_;
int color_type_;
std::unique_ptr<LibpngData> ld_;
std::unique_ptr<PngBufferReaderState> buffer_reader_state_;
};
// ----------------------------------------------------------------------------------------
template <
typename image_type
>
void load_png (
image_type& image,
const std::string& file_name
)
{
png_loader(file_name).get_image(image);
}
template <
typename image_type
>
void load_png (
image_type& image,
const unsigned char* image_buffer,
size_t buffer_size
)
{
png_loader(image_buffer, buffer_size).get_image(image);
}
template <
typename image_type
>
void load_png (
image_type& image,
const char* image_buffer,
size_t buffer_size
)
{
png_loader(reinterpret_cast<const unsigned char*>(image_buffer), buffer_size).get_image(image);
}
// ----------------------------------------------------------------------------------------
}
#ifdef NO_MAKEFILE
#include "png_loader.cpp"
#endif
#endif // DLIB_PNG_IMPORT