/*
* ofCvImage.h
* by stefanix
*
* This is the common abstract base class of all the different
* image types used in this addon. Methods with "= 0" at the end
* are pure virtual and need to be implemented by the inheriting
* image calss.
*
*/


#ifndef OF_CV_IMAGE_H
#define OF_CV_IMAGE_H


#include "ofMain.h"
#include "ofCvConstants.h"


class ofCvGrayscaleImage;
class ofCvColorImage;
class ofCvFloatImage;



class ofCvImage {


  public:
  
    int width;
    int height;

    
    ofCvImage();
    virtual ~ofCvImage();
    virtual void allocate( int w, int h ) = 0;  // pure virtual, must be
                                                // implemented by subclass
    virtual void clear();
    virtual void setUseTexture( bool bUse );    

    
    
    // Set Pixel Data - Scalars
    //
    //
    virtual void set( int value );
    virtual void operator -= ( float scalar );
    virtual void operator += ( float scalar );


    // Set Pixel Data - Arrays
    //
    //    
    virtual void operator = ( ofCvGrayscaleImage& mom ) = 0;
    virtual void operator = ( ofCvColorImage& mom ) = 0;
    virtual void operator = ( ofCvFloatImage& mom ) = 0;
    

    // Get Pixel Data
    //
    //
    virtual IplImage*  getCvImage() const { return cvImage; };    
    


    // Draw Image
    //
    //
    virtual void draw( float x, float y ) = 0;
    virtual void draw( float x, float y, float w, float h ) = 0;



    // Image Filter Operations
    //
    //
    virtual void erode( );  // based on 3x3 shape
    virtual void dilate( );  // based on 3x3 shape
    virtual void blur( int value=3 );  // value = x*2+1, where x is an integer
    virtual void blurGaussian( int value=3 );  // value = x*2+1, where x is an integer
    


    // Image Transformation Operations
    //
    //
    //virtual void setROI( int w, int h );  //TODO, support image regions
    virtual void mirror( bool bFlipVertically, bool bFlipHorizontally );
    
    virtual void translate( float x, float y );
    virtual void rotate( float angle, float centerX, float centerY );
    virtual void scale( float scaleX, float sclaeY );
    virtual void transform( float angle, float centerX, float centerY, 
                            float scaleX, float scaleY,
                            float moveX, float moveY ); 
    /**
    * undistort Usage Example: 
    * undistort( 0, 1, 0, 0, 200, 200, cwidth/2, cheight/2 );
    * creates kind of an old TV monitor distortion.
    */
    virtual void undistort( float radialDistX, float radialDistY,
                            float tangentDistX, float tangentDistY,
                            float focalX, float focalY,
                            float centerX, float centerY );
                            
    virtual void remap( IplImage* mapX, IplImage* mapY );

    virtual void warpPerspective( const ofPoint& A,
                                  const ofPoint& B,
                                  const ofPoint& C,
                                  const ofPoint& D );
    virtual void warpIntoMe( const ofCvGrayscaleImage& mom, 
                             ofPoint src[4], ofPoint dst[4] );



    // Other Image Operations
    //
    // 
    virtual int countNonZeroInRegion( int x, int y, int w, int h ) const;




  protected:
    
    virtual void swapTemp();  // swap cvImageTemp back 
                              // to cvImage after an image operation

    IplImage*  cvImage;
    IplImage*  cvImageTemp;   // this is typically swapped back into cvImage 
                              // after an image operation with swapImage()

    unsigned char* 	pixels;	  // not width stepped
    ofTexture  tex;		      // internal tex
    bool bUseTexture;
               
};


#endif
