// // ZoneMinder Image Class Interface, $Date$, $Revision$ // Copyright (C) 2003 Philip Coombes // // This program is free software; you can redistribute it and/or // modify it under the terms of the GNU General Public License // as published by the Free Software Foundation; either version 2 // of the License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. // #ifndef ZM_IMAGE_H #define ZM_IMAGE_H #include #include #include #include #include #include #include extern "C" { #if !HAVE_DECL_ROUND double round(double); #endif #include "zm_jpeg.h" } #include "zm_rgb.h" #include "zm_coord.h" #include "zm_box.h" // // This is image class, and represents a frame captured from a // camera in raw form. // class Image { protected: enum { CHAR_HEIGHT=11, CHAR_WIDTH=6 }; typedef unsigned char BlendTable[256][256]; typedef BlendTable *BlendTablePtr; protected: static bool initialised; static bool y_image_deltas; static bool fast_image_blends; static bool colour_jpeg_files; static unsigned char *abs_table; static unsigned char *y_r_table; static unsigned char *y_g_table; static unsigned char *y_b_table; static BlendTablePtr blend_tables[101]; protected: int width; int height; int colours; int size; JSAMPLE *buffer; bool our_buffer; protected: mutable unsigned int *blend_buffer; protected: static void Initialise(); static BlendTablePtr GetBlendTable( int ); public: Image() { if ( !initialised ) Initialise(); width = 0; height = 0; colours = 0; size = 0; our_buffer = true; buffer = 0; blend_buffer = 0; } Image( const char *filename ) { if ( !initialised ) Initialise(); buffer = 0; ReadJpeg( filename ); our_buffer = true; blend_buffer = 0; } Image( int p_width, int p_height, int p_colours, JSAMPLE *p_buffer=0 ) { if ( !initialised ) Initialise(); width = p_width; height = p_height; colours = p_colours; size = width*height*colours; if ( p_buffer ) { our_buffer = false; buffer = p_buffer; } else { our_buffer = true; buffer = new JSAMPLE[size]; memset( buffer, 0, size ); } blend_buffer = 0; } Image( const Image &p_image ) { if ( !initialised ) Initialise(); width = p_image.width; height = p_image.height; colours = p_image.colours; size = p_image.size; buffer = new JSAMPLE[size]; memcpy( buffer, p_image.buffer, size ); our_buffer = true; blend_buffer = 0; } ~Image() { if ( our_buffer ) { delete[] buffer; } delete[] blend_buffer; } inline int Width() const { return( width ); } inline int Height() const { return( height ); } inline int Size() const { return( size ); } inline int Colours() const { return( colours ); } inline JSAMPLE *Buffer() const { return( buffer ); } inline JSAMPLE *Buffer( unsigned int x, unsigned int y= 0 ) const { return( &buffer[colours*((y*width)+x)] ); } inline void Assign( int p_width, int p_height, int p_colours, unsigned char *new_buffer ) { if ( p_width != width || p_height != height || p_colours != colours ) { width = p_width; height = p_height; colours = p_colours; int new_size = width*height*colours; if ( size != new_size ) { size = new_size; delete[] buffer; buffer = new JSAMPLE[size]; memset( buffer, 0, size ); } } memcpy( buffer, new_buffer, size ); } inline void Assign( const Image &image ) { if ( image.width != width || image.height != height || image.colours != colours ) { width = image.width; height = image.height; colours = image.colours; int new_size = width*height*colours; if ( size != new_size ) { size = new_size; delete[] buffer; buffer = new JSAMPLE[size]; memset( buffer, 0, size ); } } memcpy( buffer, image.buffer, size ); } inline void CopyBuffer( const Image &image ) { assert( width == image.width && height == image.height && colours == image.colours ); memcpy( buffer, image.buffer, size ); } inline Image &operator=( const unsigned char *new_buffer ) { memcpy( buffer, new_buffer, size ); return( *this ); } void ReadJpeg( const char *filename ); void WriteJpeg( const char *filename ) const; void DecodeJpeg( JOCTET *inbuffer, int inbuffer_size ); void EncodeJpeg( JOCTET *outbuffer, int *outbuffer_size ) const; void Overlay( const Image &image ); void Blend( const Image &image, int transparency=10 ) const; static Image *Merge( int n_images, Image *images[] ); static Image *Merge( int n_images, Image *images[], double weight ); static Image *Highlight( int n_images, Image *images[], const Rgb threshold=RGB_BLACK, const Rgb ref_colour=RGB_RED ); Image *Delta( const Image &image ) const; void Annotate( const char *text, const Coord &coord, const Rgb colour ); void Annotate( const char *text, const Coord &coord ); Image *HighlightEdges( Rgb colour, const Box *limits=0 ); void Timestamp( const char *label, const time_t when, const Coord &coord ); void Colourise(); void DeColourise(); void Clear() { memset( buffer, 0, size ); } void Fill( Rgb colour, const Box *limits=0 ); void Hatch( Rgb colour, const Box *limits=0 ); void Rotate( int angle ); void Scale( unsigned int factor ); }; #endif // ZM_IMAGE_H