Introduction
I made a simple but nice photo editor with OpenCV. It can be used to edit photos, such as, adding simple effects, changing the brightness and the contrast, histogram equalization, face detection, etc. For the GUI, I use wxWidgets so it can be easily ported to the Linux operating system.
Background
OpenCV is an open source computer vision library in C/C++. It provides us many APIs for image data manipulation. It is OS independent and optimized for real-time applications. Go to the OpenCV website and download the latest version. Install and check sample directory. Find some tutorials and enjoy.
wxWidgets is an OS independent open source GUI library. By using this library we can make an OS independent application. Check here at the wxWidgets website.
The Plan
Here is the plan. We are going to make a photo editor software that is capable of:
- Converting an image to grayscale and black and white.
- Do per channel adjustment of brightness and contrast based on its color space.
- Change color space: RGB, YCrCb, HSV, and HLS.
- Rotate, flip, or resize an image.
- Do histogram equalization.
- Show histogram graph.
- Detect faces.
- Generate erode and dilate effects.
- Have "UNDO" capability.
The Code
The class for image manipulation is allocated in files mango.h and mango.cpp with name CMango
.
class CMango
{
public:
enum type{
IM_AUTOMATIC,
IM_GRAYSCALE,
IM_BLACKNWHITE,
IM_RGB,
IM_HSV,
IM_HLS,
IM_YCrCb,
IM_ZOOMIN,
IM_ZOOMOUT,
IM_HORIZONTAL = 25,
IM_VERTICAL,
IM_DILATE = 40,
IM_ERODE,
IM_FACE1,
};
CMango(void);
~CMango(void);
int LoadImageFromFile(char *FileName);
void ShowActiveImage(char *WindowName);
int SetProperty();
std::string GetProperty();
void LoadPrestoredImage(int type);
void PushImage(IplImage *Image){ImageList.push_front(Image);};
void PopImage(){ if (!ImageList.empty()) ImageList.pop_front();};
int SaveToFile(char *FileName);
void Rotate(int angle);
void ZoomImage(int flag);
int GetImageElement(int channel, int brightness, int contrast);
void FlipImage(int flag);
void ApplyEffects(int flag);
void FaceDetect(int flag);
void ShowHistogram();
int HistogramEqualization();
int ChangeColorSpace(int flag);
private:
static IplImage *Automatic;
static IplImage *Blacknwhite;
static IplImage *Grayscale;
static IplImage *Buf;
static std::list<IplImage *> ImageList;
static std::list<std::string> PropertyList;
static int ColorSpace;
};
When an image is loaded, it is kept in data member Automatic
. Actually we have three pre-stored images here: Automatic
, Blacknwhite
, and Grayscale
. Every operation we perform will push an active image (image that is being displayed) to ImageList
so we can perform "undo." The "undo" command basically will popup an image from ImageList
to be displayed.
I also did some experiments on color spaces. Adjusting the per-channel brightness and contrast of an image can give us an awesome effect depending on its color spaces. I used an algorithm that I found in this site. Since OpenCV will only put RGB on display, I have to do these: Convert image to target color space, do the adjustment that I want, and finally, convert it back to RGB to be displayed.
Unfortunately, I encountered a problem. The problem is that the algorithm is for RGB color space. I can't just use it for other color spaces. But, take a look at this:
|
|
Range |
RGB |
R |
0 - 1 |
|
G |
0 - 1 |
|
B |
0 - 1 |
YCrCb |
Y |
0 - 1 |
|
Cr |
-0.5 - 0.5 |
|
Cb |
-0.5 - 0.5 |
HSV |
H |
0-360 |
|
S |
0 - 1 |
|
V |
0 - 1 |
HLS |
H |
0-360 |
|
L |
0 - 1 |
|
S |
0 - 1 |
Based on this table I think it is possible to apply the same algorithm for other color spaces besides RGB as long as they are in the same range.
Please notice that I only implemented the histogram curve for RGB color space only. If you change the color space, the histogram curve is no longer valid because of difference in range.
Points of Interest
I implemented many basic functions of OpenCV here. I hope It might be useful for you to start learning about OpenCV.
References