Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/yashk2000/image-processing

A set of algorithms and other cool things that I learned while doing image processing with openCV using C++ and python.
https://github.com/yashk2000/image-processing

cpp detecting-edges detecting-faces edge-detection histogram-equalization image-inpainting image-processing opencv opencv-python python3 superpixels thresholding video-player

Last synced: 2 months ago
JSON representation

A set of algorithms and other cool things that I learned while doing image processing with openCV using C++ and python.

Awesome Lists containing this project

README

        

# Contents

1. [Opening an image using openCV](https://github.com/yashk2000/Image-Processing#1-opening-an-image-using-opencv)
2. [Opening a video using openCV](https://github.com/yashk2000/Image-Processing#2-opening-a-video-using-opencv)
3. [Creating a video player with play, pause and a seekbar](https://github.com/yashk2000/Image-Processing#3-creating-a-video-player-with-play-pause-and-a-seekbar)
4. [A simple transformation: Smoothing an image](https://github.com/yashk2000/Image-Processing#4-a-simple-transformation-smoothing-an-image)
5. [Downsampling an image](https://github.com/yashk2000/Image-Processing#5-downsampling-an-image)
6. [Simple way to detect edges of an image using Canny Edge Detector](https://github.com/yashk2000/Image-Processing#6-simple-way-to-detect-edges-of-an-image-using-canny-edge-detector)
7. [Taking input from device camera](https://github.com/yashk2000/Image-Processing#7-taking-input-from-device-camera)
8. [Getting and setting pixels in an image](https://github.com/yashk2000/Image-Processing#8-getting-and-setting-pixels-in-an-image)
9. [Reading a video and writing it to a file](https://github.com/yashk2000/Image-Processing#9-reading-a-video-and-writing-it-to-a-file)
10. [Reading and writing a video recorded from camera to a file](https://github.com/yashk2000/Image-Processing/blob/master/README.md#10-reading-and-writing-a-video-recorded-from-camera-to-a-file)
11. [Data Types in openCV](https://github.com/yashk2000/Image-Processing#11-data-types-used-in-opencv)
12. [Creating a histogram with the frequency of pixels in an image](https://github.com/yashk2000/Image-Processing#creating-a-histogram-with-the-frequency-of-pixels-in-an-image)
13. [Enhance an image using Histogram Equalization](https://github.com/yashk2000/Image-Processing#enhance-an-image-using-histogram-equalization)
14. [Constrast-Limited Adaptive Histogram Equalization](https://github.com/yashk2000/Image-Processing#constrast-limited-adaptive-histogram-equalization)
15. [Some basics of openCV with python](https://github.com/yashk2000/Image-Processing#15-basics-of-image-processing-with-python)
16. [Detecting faces in images](https://github.com/yashk2000/Image-Processing#16-detecting-faces-in-images)
17. [Detecting faces in videos](https://github.com/yashk2000/Image-Processing#17-detecting-faces-in-videos)
18. [Thresholding](https://github.com/yashk2000/Image-Processing#18-thresholding)
19. [Blur Detection](https://github.com/yashk2000/Image-Processing#19-blur-detection)
20. [Detecting the centre of a geometric shape](https://github.com/yashk2000/Image-Processing#20-detecting-the-centre-of-a-geometric-shape)
21. [Long Exposure](https://github.com/yashk2000/Image-Processing#21-long-exposure)
22. [Segmentation into Superpixels](https://github.com/yashk2000/Image-Processing#22-segmentation-into-superpixels)
23. [Colors in superpixels](https://github.com/yashk2000/Image-Processing#23-colors-in-superpixels)
24. [Image Inpainting using OpenCV](https://github.com/yashk2000/Image-Processing/blob/master/README.md#24-image-inpainting-using-opencv)

# 1) Opening an image using openCV

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/openImage.cpp)

After compilation when the executable is run from the termilnal, provide a path to the place where the image is stored in the system. The image is displayed on the screen till the user presses any key.

Image is opened by the command `cv::imread(argv[1], -1);`. This can open an image of BMP, JPEG, PNG, PPM, RAS, etc formats and returns the image in a `cv::Mat` format. The image is opened by `cv::namedWindow`. It opens a window with the name given as a parameter to the function. This functions and the window which opens is provided by the **HighGUI Library**. The `WINDOW_AUTOSIZE` parameter automatically gives the window such a size as to accomodate the true size of the image. Now to display the image on the window, the `cv::imshow()` function is called. The `cv::waitKey()` function will paause the program execution and will wait until the user presses a key, at which point it will resume execution, hence calling the `cv::destroyWindow` function and closing the image.

Here's how the window with the photo will appear:(it might appear differently in different systems depending upon the photo)

![Screenshot_20190818_160716](https://user-images.githubusercontent.com/41234408/63223432-e1607a00-c1d2-11e9-9bb6-c4d4003b5ea6.png)

# 2) Opening a video using openCV

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/openVid.cpp)

The video can be opened and closed the same as the image in the previous section.

The object `cap` of the type `cv::VideoCapture` is instatiated and is used to open or close videos. `cv::Mat` is used to create an object called `frame`. This object holds the frames of the video. The video is displayed frame by frame until there is no frame left which is checked by `frame.empty()`.

Here's how the most basic video player with HighGUI will look:

![Screenshot_20190818_005927](https://user-images.githubusercontent.com/41234408/63216547-4f199100-c154-11e9-8b2b-2dd79dc6380d.png)

# 3) Creating a video player with play, pause and a seekbar

The source code for creating a video player can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/vidPlayer.cpp)

Here we use global variables to keep track of the trackbar position and update in when needed, along with the `cv::VideoCapture` object as global. The `run` variable helps in keeping displaying frames as long as it isn't 0. We update the position of the seekbar with every frame being displayed. But this will put us into **single-step** mode whoch we don't want. To avoid this, we use the `dontset` variable to help us stay in the **run** mode.

```cpp

void onTrackbarSlide(int pos, void *) {
cap.set(cv::CAP_PROP_POS_FRAMES, pos);
if( !dontset )
run = 1;
dontset = 0;
}
```
We need the number of frames in the video in order to calliberate the seekbar. This we get by using the `cap.get(cv::CAP_PROP_FRAMECOUNT`. Next, using thr number of frames, and also setting a variable to get the position of the seekbar, we use `cv::createTrackbar` to create the trackbar.

```cpp

createTrackbar("Position", "Video", &slider_position, frames, onTrackbarSlide);
```

While the video is running, we keep taking user input, inside the while loop. The character variable, `ch` is used for this. If user enters `r`, the video enters **run** mode and plays like nay normal video. On pressing `s`, we enter the **single-step** mode. If the video is in run mode and then we press `s`, the video gets paused. Hence the pause functionality can be implemented this way too. If the user presses `h`, we display the user with options of what can be done:
`Press r to run the video, s to pause or run the video frame by frame, h for help and esc to quit`.

This way, just using c++, we can use the HighGUI package provided in openCV to create our own video player.

This is how the video player looks with the seekbar:

![Screenshot_20190818_010241](https://user-images.githubusercontent.com/41234408/63216703-fe576780-c156-11e9-9a88-19dbad8e4b42.png)

And here's how the terminal will be looking like while you use `r` or `s` or `h` or `esc` to control the player:

![Screenshot_20190818_010249](https://user-images.githubusercontent.com/41234408/63216712-1b8c3600-c157-11e9-8128-7b1477c1d0a0.png)

# 4) A simple transformation: Smoothing an image

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/imgSmoothing.cpp)

Here we will be smoothing an image using Gaussian Blur. Image smoothing basically refers to remove the sharp edges of an image. The resulting image is usually a bit more blurred that the input image. This tranformation is pretty simple. We take in an image as input in the same way as we did while opeing an image [earlier](https://github.com/yashk2000/Image-Processing#1-opening-an-image-using-opencv). After this, we create 2 windows, one for the initial image, and the other to display the smoothened image. We can smoothen the image by simply calling the `cv::GaussianBlur()` function. The Gaussain Blur function always takes in odd numbers as input. Here we have blurred the input image with a 5X5 Gaussian Blur convolution matrix. We have called the `cv::GaussianBlur()` function twice, so as to give us more pronounced effect. Here instead of Gaussian Blur, we can use other library methods such as `cv::blur()`, `cv::medianBlur()` or `bilateralFilter()`.

**Image before smoothing:**
![Screenshot_20190818_160716](https://user-images.githubusercontent.com/41234408/63223432-e1607a00-c1d2-11e9-9bb6-c4d4003b5ea6.png)

**Image after smoothing:**
![Screenshot_20190818_160734](https://user-images.githubusercontent.com/41234408/63223434-e4f40100-c1d2-11e9-9472-c5491f7c11d8.png)

## Other methods for image smoothing

### blur()

We can use the `cv::blur()` method in [this](https://github.com/yashk2000/Image-Processing/blob/master/imgSmoothingBlur.cpp) way.

**Image after blurring**
![Screenshot_20190818_165436](https://user-images.githubusercontent.com/41234408/63223842-29829b00-c1d9-11e9-8d3f-4e02883a9a8a.png)

### medianBlur()

We can use the `cv::medianblur()` method in [this](https://github.com/yashk2000/Image-Processing/blob/master/imageSmoothingMedianBlur.cpp) way.

**Image after blurring**
![Screenshot_20190818_165500](https://user-images.githubusercontent.com/41234408/63223833-07891880-c1d9-11e9-8de9-1e37b4790da1.png)

### bilateralFilter()

We can use the `cv::bilateralFilter()` method in [this](https://github.com/yashk2000/Image-Processing/blob/master/imgSmoothingBilateralFilter.cpp) way.

In bilateralFilter(), we need to keep the input and output images differnet, else we get an error.

Bilateral filter also takes a gaussian filter in space, but one more gaussian filter which is a function of pixel difference. Gaussian function of space make sure only nearby pixels are considered for blurring while gaussian function of intensity difference make sure only those pixels with similar intensity to central pixel is considered for blurring. So it preserves the edges since pixels at edges will have large intensity variation. The texture of the image is reduced, but the edges are conserved.

**Image after blurring**
![Screenshot_20190818_165158](https://user-images.githubusercontent.com/41234408/63223805-9e090a00-c1d8-11e9-8c8b-da43f3f75df0.png)

# 5) Downsampling an image

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/DownSample.cpp)

This is one of the applications of Gaussian Blurring. We use Gaussian Blurring to downsample an image to change the scale in which an image is viewed. We use the `cv::pyrDown()` function to do both Gaussian Blurring and downsampling simultaneously.

**Result of Downsampling an image:**
![Screenshot_20190820_001735](https://user-images.githubusercontent.com/41234408/63291423-1b1fa680-c2e1-11e9-8165-c07b8e5f352d.png)

The first image is the input image, which is larger. The second one is the output image which is half the size of the the input image. This shows that the image was downscaled by a margin of 2.

# 6) Simple way to detect edges of an image using Canny Edge Detector

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/CannyDetector.cpp)

To detect image edges, we use the Canny Edge Detector. To do this, we first convert the image to grey scale using `cv::cvtColor()`. This is done because Canny Egde Detector needs only a sigle channel to write to and a grey scale image is a single-channel image. Once the image is in grey scale, we can proceed with edge detection using `cv::Canny`. This might work even without converting the image to grey scale, but since Canny is a **single-channel** writing function, it is recommended to create a single-channel image i.e. a grey scale image.

**The grey scale and the final image with edges highlight will look like the following:**

**Input Image**
![Screenshot_20190820_005900](https://user-images.githubusercontent.com/41234408/63294985-41494480-c2e9-11e9-997f-89eb1337c1f3.png)

**Grey Scale Image**
![Screenshot_20190820_005909](https://user-images.githubusercontent.com/41234408/63295000-49a17f80-c2e9-11e9-9fad-1a1a676db1c4.png)

**Canny Image with prominent edges**
![Screenshot_20190820_005918](https://user-images.githubusercontent.com/41234408/63295015-51612400-c2e9-11e9-83ab-87cab8973566.png)

**Canny image when detecting edges without converting to grey scale**
![Screenshot_20190820_011012](https://user-images.githubusercontent.com/41234408/63295042-5920c880-c2e9-11e9-98da-a37f12e0e3de.png)

Here you can see that some details are missing. It also shows some edges which are actually not image edges. The Canny image in which edges were detected after converting to grey scale has more details. That's the reason for using a single-channel image while doing edge detection using Canny.

# 7) Taking input from device camera

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/InputFromCamera.cpp)

Here, if we provide a path as input, the video at the provided path will be opened. Otherwise, the camera will open and start taking a video. If there are no input arguments, we use the `VideoCapture` object to open the camera. If there is only one camera conected to the device, then we pass 0 as a parameter to the `VideoCapture` object. If there are multiple cameras connected, we can pass -1 as the parameter to open any one camera randomly.
The video is displayed in the same way as a video stored in the computer. We use the same process as given in [this](https://github.com/yashk2000/Image-Processing/blob/master/openVid.cpp) code.

Now while displaying the video, we can put any effect we want. It is displayed frame by frame. So we can treat each frame as an image and put any effect we want on the frame. For Example:

**Normal Code**
```cpp

while (true) {
cap >> frame;
if( frame.empty() ) break;
cv::imshow( "Example3", frame );
if( cv::waitKey(33) >= 0 ) break;
}
```

**Code with grey scale effect while displaying video**
```cpp

while (true) {
cap >> frame;
if( frame.empty() ) break;
cv::cvtColor( frame, output_frame, cv::COLOR_BGR2GRAY);
cv::imshow( "Output", output_frame );
if( cv::waitKey(33) >= 0 ) break;
}
```
This will display your video in greyscale format.

Here's how it looks while displaying a video:

**Camera Check**

![Screenshot_20190820_1732571](https://user-images.githubusercontent.com/41234408/63347080-06402300-c374-11e9-8766-d8279b7f4a20.png)

**Normal Video being recorded by camera is displayed**

![Screenshot_20190820_173257](https://user-images.githubusercontent.com/41234408/63347062-f4f71680-c373-11e9-8be7-5bcff601c798.png)

**Grey Scaled Video**

![Screenshot_20190820_180003](https://user-images.githubusercontent.com/41234408/63347331-7b135d00-c374-11e9-8fe6-413a83a906d2.png)

# 8) Getting and setting pixels in an image

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/pixels.cpp)

Here we use the Vec3b object to store the values of Red, green and blue colored pixels in a RGB image. Vec3b represnts a vector with 3 byte enteries, with each byte representing one color.

If we are using an image with a single color, we can simply get the number of pixels of that color in the image by doing :
```cpp

(unsigned int)image.at(y, x) //x, y represent pixel coordinates
```

To set a particular pixel of an image to a particular color, we simply access the pixel using it's coordinates and set it to the desired value.
```cpp

image.at(x, y) = 128;
```

# 9) Reading a video and writing it to a file

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/WriteFile.cpp)

This is essentailly just an extension to the [code](https://github.com/yashk2000/Image-Processing/blob/master/openVid.cpp) where we opened an existing video file from the local storage of the computer. Here we just use the `cv::VideoWriter` to write our desired frames to the file we want. We specify that we are writing a video in the common MJPG(motion jpeg) by using the `CV_FOURCC` codec. Here, we have to give the `writer` object some parameters. The first parameter is the output file path, the second one is the output file type, then comes the number of frames and finally the dimensions(size) of the output frame/frames.

# 10) Reading and writing a video recorded from camera to a file

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/ReadWriteLogPolarFromCamera.cpp)

This program is just an extension of the [one](https://github.com/yashk2000/Image-Processing/blob/master/InputFromCamera.cpp) where we took input from camera and [one](https://github.com/yashk2000/Image-Processing/blob/master/WriteFile.cpp) in which we wrote a video to another file. Here we just have to give the path to the output file when we run the code, instead of a path to the input file.

There's just one more creative thing we did here. Instead of directly writing the output to a file, which will be way easier, we first converted the camera input to **log polar** form and wrote that to an output file. Log polar form is something like the way our eyes actually process what they see. We'll go into the details of the log polar form later on.

# 11) Data types used in openCV

Here we will be noticing many of the data types ending with 2d, 2i, 2f, 3d, 3i, 3b and so on. Here the number basically represents the dimension of the data stored and the characters indicate the following: `i for integer`, `f for float`, `d for double` and `b for unsigned character`.

### cv::Vec<>

The `cv::Vec<>` is used as a container for almost any type of data(including objects, pointers etc.) in cpp. But we mainly use it as a container for primitive data types such as int, float, char etc.
We will not be using the `cv::Vec<>` template much. Instead we use the aliases that exist for some common instantiations of `cv::Vec<>`. Some of these are:
- `cv::Vec2i` - This will serve as a two element integer vector
- `cv::Vec3i` - This will serve as a three element integer vector
- `cv::Vec4d` - This will serve as a four element double precision floating point vector

The limitation of the cv::Vec class is that it is not very effecient for handling large arrays. For that we have the `cv::Mat` class(which we will learn about later on).

| Operations | Examples |
| --- | --- |
| Default constructor | Vec2s v2s; Vec6f v6f; |
| Member access | v4f[i], v3w(j); // both [], () are fine
| Vector cross product | v3f.cross(u3f); |

### cv::Matx<>

The `cv::Matx<>` class is is a fixed matrix class which is highly effecient for dealing with 2 x 2, 3 x 3 and 4 x 4 matrix operations. Here too the only drawback is that we need to know the matrix dimensions before hand, and this is not efficient when it comes to handling large sized arrays. Again thr `cv::Mat` class(which wiil be discussed later on) comes to the rescue here.

If you want a mtrix with dimensions other than the specified ones, we can initialize it ourselves -> `cv::Matx<5, 5, float>`

| Operations | Examples |
| --- | --- |
| Default Constructor | cv::Matx33f m33f;
| | cv::Matx43d m43d; |
| Matrix of identical elements | m33f = cv::Matx33f::all(x); |
| Matrix of zeros | m230 = cv::Matx23d::zeros(); |
| Matrix of ones | m161 = cv::Matx16d::ones(); |
| Create a unit matrix | m33f = cv::Mat33f::eye(); |
| Matrix that can hold the diagnols of another matrix | m33f = cv::Matx33f::diag(); |
| Create a matrix with uniformly distributed entries | m44f = cv::Matx44f::randu(min, max); |
| Create a matrix with normally distributed entries | m44f = cv::Matx44f::nrandn(mean, varience); |
| Member access | m(i, j)//for multi dimensionsal |
| | m(i) //for oone dimnesional matrices only |
| Matrix algebra | m1 x m2, m1 + m2, m1 - m2 |
| Singleton algebra | m x a, a x m, m / a |
| Comparison | m1 == m2; |
| Dot Product | m1.dot(m2) |
| | m1.ddot(m2) |
| Extract a 2x2 matrix at position i, j | m44f.get_minor<2, 2>(i, j); |
| Extract row i | m14f = m44f.row(i); |
| Extract column j | m41f = m44f.col(j); |
| Extract diagnol | m41f = m44f.diag(); |
| Find traspose | n44f = m44f.t(); |
| Inverse of a matrix | n44f = m44f.inv(method) // default method is `cv::DECOMP_LU` |
| Per element multiplication | m1.mul(m2) |

### cv::Point<>

This class is similar to the `cv::Vec<>` class. We can store 2 to 3 primitive data types using instances of `cv::Point<>`. Here we can call the data stored using the dot operator and class insatance itself.
For example if the instance is `point`. For a vector, we'll do `point[0]`, `point[1]` and so on. Instead in the `cv::Point<>` class, we simply do, `point.x`, `point.y` and so on.
Same as the `cv::Vec<>` class, `cv::Point<>` class also contains some aliases. The most commonly used ones are:
- `cv::Point2i`
- `cv::Point2f`
- `cv::Point2d`
- `cv::Point3i`
- `cv::Point3f`
- `cv::Point3d`

| Operation | Example |
| --- | --- |
| Default Constructor | cv::point 2i p; |
| | cv::point 3f p; |
| Member access| p.x, p.y, p.z(if 3 dimensional) |
| Dot Product | float x = p1.dot(p2) |
| Double Precision Dot Product | double x = p1.ddot(p2) |
| Cross Product | p1.cross(p2) |
| To check if point p lies inside rectangle r | p.inside(r) |

### cv::Scalar<>

- Used for storing a four dimensional point.
- It aliased to store a four component vector with double precision components.
- Data is called by objects and not the dot operator.
- Directly derived from the `cv::Vec<>` class( `cv::Vec

- Has two members: height and width
- It is an alias for `cv:Size2i`
- To store floating point data, use `cv::Size2f`

| Operations | Examples |
| --- | --- |
| Default construtor | cv::Size s;
| | cv::Size2i s; |
| | cv::Size2f s;|
| Member access | s.width, s.height|
| Finding area| s.area() |

### cv::Rect<>

- Has four members: height, width, x and y
- Alias to store the ineger form of a rectangle

| Operations | Examples |
| --- | --- |
| Default Constructor | cv::Rect r |
| Construct from origin and size | cv::Rect(p, sz) |
| Construct from 2 given points | cv::Rect(p1, p2) |
| Member Access | r.x, r.y, r.width, r.height |
| Area | r.area() |
| Finding coordinates of the upper left corner | r.tl() |
| Finding coordinates of the bottom right corner | r.br() |
| Checking if point p is inside rectagle r | r.contains(p) |
| Intersection of rectangle r1 and r2 | cv::Rect r3 = r1 \& r2; |
| | r1 \&= r2; |
| Minimum area rectangle containing r1 and r2 | cv::Rect r3 = r1 \| r2; |
| | r1 \|= r2; |
| Tranalte a rectan;ge by amount x | cv::Rect rx = r + x; |
| | r += x; |
| Enlarge a rectanlge by s | cv::Rect rs = r + s; |
| | r += s; |
| Compare rectangles for equality | bool e = (r1 == r2); |

### cv::RotatedRect<>

- Store a rectangle which is not axis aligned
- Contains:

- `cv::Point2f` object, called `center`

- `cv::Size2f` object, called `size`

- A float called `angle`


| Operations | Examples |
| --- | --- |
| Default Constructor | cv::RotatedRect rr(); |
| Construct a rectangle from 2 points | cv::RotatedTect(p1, p2); |
| Construct a rectangle using a point, size and angle | cv::RotatedRect(p, s, theta); |
| Member access | rr.center, rr.size, rr.angle |
| Return all corners | rr.points(pts[4]) |

### cv::Complexf<>

| Operations | Examples |
| --- | --- |
| Default constructor | cv::Complexf z1 |
| | cv::Complexd z2 |
| Member access | z.re, z.im; |
| Complex Conjugate | z2 = z1.conj(); |

# Creating a histogram with the frequency of pixels in an image

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/histogram.cpp)

This code just generates a graph with the frequency of pixels in an image.

**Input Image**

![kakashi](https://user-images.githubusercontent.com/41234408/63835160-7cf0a800-c994-11e9-9451-1b9a5bcb0189.jpg)

**Histogram**

![Screenshot_20190828_130431](https://user-images.githubusercontent.com/41234408/63835216-9f82c100-c994-11e9-8760-9f486592b2ed.png)

# Enhance an image using Histogram Equalization

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/histogramEqualization.cpp)

[Here](https://github.com/yashk2000/Image-Processing/blob/master/hisEqualMyAlgo.cpp) is my own implementation of Histogram Equalization instead of using the function directly provided by openCV.

It is a method that improves the contrast in an image, in order to stretch out the intensity range. Equalization implies mapping one distribution (the given histogram) to another distribution (a wider and more uniform distribution of intensity values) so the intensity values are spread over the whole range.

**Original Image**

![input](https://user-images.githubusercontent.com/41234408/63831243-d99b9500-c98b-11e9-87f0-aa6fb5945fe9.png)

**Histogram of original image**

![hist](https://user-images.githubusercontent.com/41234408/63831240-d7393b00-c98b-11e9-895f-aaf51ff6ddbd.png)

**Output Image**

![output](https://user-images.githubusercontent.com/41234408/63831322-09e33380-c98c-11e9-83fa-ef5c8795e1a6.png)

**Histogram of output image**

![Screenshot_20190828_120113](https://user-images.githubusercontent.com/41234408/63831347-19fb1300-c98c-11e9-9e03-b1fcb32c1c09.png)

# 14) Constrast-Limited Adaptive Histogram Equalization

The code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/clahe.cpp)

Histogram equalization is a global function, where as CLAHE is a local function. In CLAHE, a small block of an image is taken, on that small block histogram equalization is performed. In CLAHE, we have two factors: block size and clip limit based on which histogram equalization is performed on each block.

**Original Image**

![Screenshot_20190919_220656](https://user-images.githubusercontent.com/41234408/65263635-aa131f00-db2a-11e9-8f7a-97c459fd0db6.png)

**Histogram Equlaized Image**

![Screenshot_20190919_2206561](https://user-images.githubusercontent.com/41234408/65263684-c1520c80-db2a-11e9-9147-5ad5154f4e9a.png)

**CLAHE Image**

![Screenshot_20190919_220830](https://user-images.githubusercontent.com/41234408/65263712-cca53800-db2a-11e9-8d96-871001233789.png)

# 15) Basics of image processing with python

The code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/opencv_tutorial_01.py).

Two packages have been used here, OpenCV and imutils. imutils is another image processing library which has a lot of useful helper functions.

- Reading an image: cv2.imread("path to image")
- Finding dimensions of an image: (h, w, d) = image.shape
This will return the height, width and depth of an image. Depth is the number of channels in an image, which is 3 for a RGB image.
- Accessing individual pixels: (B, G, R) = image[y, x]
This will return the intensity of Blue, Green and Red colors at the pixel (x, y). The input is given in (y, x) format.
- Extracting a ROI from an image: image[startY:endY, startX:endX]
- Resizing an image witout maintaining aspect ratio:
```python
resized = cv2.resize(image, (300, 300))
```
This will resize an image to a 300x300 block without maintaining the aspect ratio.
- Resizing an image while maintaining aspect ratio:
```python
r = 300.0 / w # calculate the ratio of the new width to the old width
dim = (300, int(h * r)) # calculate the height using the ratio by multiplying h by r (the original height and our ratio respectively)
resized = cv2.resize(image, dim) # resize while maintaining the aspect ratio
```
- Using imutils to automatically resize an image while maintaining aspect ratio:
```python
resized = imutils.resize(image, width=300)
```
- Rotating an image with OpenCV and imutils:
```python
center = (w // 2, h // 2) # find the centre of the image
M = cv2.getRotationMatrix2D(center, -45, 1.0) # obtain the rotation matrix, -45 for rotating clockwise by 45 degree, 45 for rotating anticlockwise by 45 degrees
rotated = cv2.warpAffine(image, M, (w, h)) #apply affine warp to get the rotated image
```
We can instead use the `rotate` function provided by `imutils` to perform the rotation in a shorter way.
```python
rotated = imutils.rotate(image, -45)
```
But this method has a drawback. It doesn't take into consideration the change in dimensions of the image and therefore cuts of portions of the image while rotating it.

To fix that we use the `rotate_bounds` method in `imutils`
```python
rotated = imutils.rotate_bound(image, 45) #here positive angle means clockwise rotation
```

A comparison of both the functions together is done here in [this script](https://github.com/yashk2000/Image-Processing/blob/master/rotate_image.py).
- Smoothing an image: cv2.GaussianBlur(image, (11, 11), 0)

Guassian blur is performed using an 11x11 kernel.
- Drawing on an image:
```python
output = image.copy()
cv2.rectangle(output, (320, 60), (420, 160), (0, 0, 255), 2)
cv2.circle(output, (300, 150), 20, (255, 0, 0), -1)
cv2.line(output, (60, 20), (400, 200), (0, 0, 255), 5)
```
- Writing text on an image:
```python
cv2.putText(output, "Naruto rocks!!!!", (10, 25),
cv2.FONT_HERSHEY_SIMPLEX, 0.7, (0, 255, 0), 2)
```

# 16) Detecting faces in images

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/detect_faces.py).

Here, openCV's deep learning face detector has been used. Two requirements for this are the `.prototxt` and the `.caffemodel` files which have been uploaded in the repo as well.

Once the models have been loaded, we use `dnn.blobFromImage` to do the preprocessing such as setting the blob dimensions and normalization.

Next we detect the faces in an image. This is done using two simple lines oof python code:

```python
net.setInput(blob)
detections = net.forward()
```

Once this is done, we iterate over all the detections, draw boxes aroud faces which have a certain threshold of confidence(in order to filter out weaker detections which most probablt will not be faces).

This code can be run by running the following command in your terminal:
```bash
python detect_faces.py --image --prototxt --model
```

The code in action:

![Screenshot_20191110_021430](https://user-images.githubusercontent.com/41234408/68534887-0d1d7700-0360-11ea-895e-f61217bcdd6a.png)

![Screenshot_20191110_021430](https://user-images.githubusercontent.com/41234408/68534901-2faf9000-0360-11ea-9246-051585ca6799.png)

# 17) Detecting faces in videos

The source code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/detect_faces_video.py)

This is very much similar to the above piece of code in which we used the deep leanring module of openCV to detect faces in am image. The exact same technique has been used to detect faces in a video.

The difference between an image and a video is that, a video is a collection of images called the frames of a video. So we take each frame of a video, treat it like an image and perform the same technique on it as the one described above. We start taking video input using a video file or from the device webcam using `VideoStream`.

We can get the frames of a video using the following piece of python code:
```python
frame = vs.read()
```

# 18) Thresholding

Thresholding is the simplest method of segmenting image. From a grayscale image, thresholding can be used to create binary images.

There are three types of thresholding methods implemented here:
1) [Riddler Calvard](https://github.com/yashk2000/Image-Processing/blob/master/RiddlerCalvard.py)
2) [Adaptive Thresholding](https://github.com/yashk2000/Image-Processing/blob/master/adaptiveThresholding.py)
3) [Otsu Algorithm](https://github.com/yashk2000/Image-Processing/blob/master/otsu.py)

**Original Image**

![orig](https://user-images.githubusercontent.com/41234408/72064616-43488a00-3302-11ea-9b3e-0964d6edeb98.png)

**Thresholding by Riddler Calvard**

![Screenshot_20200109_165427](https://user-images.githubusercontent.com/41234408/72064626-46437a80-3302-11ea-93d0-6fe5d3230b07.png)

**Thresholding by Adaptive Thresholding method**

![Screenshot_20200109_165447](https://user-images.githubusercontent.com/41234408/72064632-480d3e00-3302-11ea-8bc9-0722d87220fe.png)

**Thresholding by otsu algorithm**

![Screenshot_20200109_165506](https://user-images.githubusercontent.com/41234408/72064634-4a6f9800-3302-11ea-93a6-8f17c3239676.png)

# 19) Blur Detection

The code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/blurDetection.py)

This code detects the amount of blurring present in an input image by using the Laplacian operator provided by openCV. We use the Laplacian operator to compute the varience of an image. A sharp image will have a high varience while a blurred image being smooth, will have a low varience.

**Example 1**

![Screenshot_20200109_172347](https://user-images.githubusercontent.com/41234408/72065753-f1edca00-3304-11ea-9d48-7b89834dc4ee.png)

**Example 2**

![Screenshot_20200109_172401](https://user-images.githubusercontent.com/41234408/72065757-f4502400-3304-11ea-938f-9cf8ea62d92e.png)

# 20) Detecting the centre of a geometric shape

The code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/shapeCentre.py).

The first step is to preprocess the image by applying grayscale conversion, Gaussian smoothing using a 5 x 5 kernel, and finally thresholding.

Then we extract the outline i.e. the contours of the image. Next we loop over each contours, and compute image moments for the contour region. Image moments are used to characterize the shape of an object in an image. These moments capture basic statistical properties of the shape, such as the area of the object, the centroid orientation, etc.
Here we use the centre of the contour to determine the image shape.

**Example 1**

![Screenshot_20200109_173635](https://user-images.githubusercontent.com/41234408/72066685-3f6b3680-3307-11ea-8349-555f3075908d.png)

**Example 2(works with multiple shapes present in an image too)**

![Screenshot_20200109_173649](https://user-images.githubusercontent.com/41234408/72066691-42febd80-3307-11ea-9897-f3b5ac6b7310.png)

**Example 3(If there are multiple shapes, centre of all the shapes can be detected)**

![Screenshot_20200109_173657](https://user-images.githubusercontent.com/41234408/72066695-42febd80-3307-11ea-8cea-bbe2bb240109.png)

# 21) Long Exposure

The code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/longExposure/longExposure.py).

Long exposure is the process of creating a photo that illustrates the effect of passing time. This will be something that is difficult to capture via regular photographic methods(unless you put in a lot of effort and some high end equipment).

This code takes in a video as input and uses frame averaging to simulate long exposures.

Fox exmaple:

Video:

![gif](https://github.com/yashk2000/Image-Processing/blob/master/longExposure/mountain.gif)

Output:

![image](https://github.com/yashk2000/Image-Processing/blob/master/longExposure/output/mountainResult.jpg)

# 22) Segmentation into Superpixels

The code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/superpixel.py).

A group of pixels extracted from the pixel grid of an image which share some characteristics are called superpixels. These regions provide a lot more information about an image than a single pixel of the image would.
Performing operation on super pixels reduces the computation effort. Once the superpixels regions are obtained, instead of dealing with hundreds of thousands of pixels, we can just deal with thousands of superpixels. This also provides more perceptual meaning than a single pixel as all pixels belonging to a superpixel group share some common characteristics.

For exmaple, here is the input image:

![photo](https://user-images.githubusercontent.com/41234408/80244661-eee5d200-8686-11ea-8fb5-db611b6ef322.jpg)

Image segemnted into 100 superpixels

![Screenshot from 2020-04-24 23-49-44](https://user-images.githubusercontent.com/41234408/80244576-cbbb2280-8686-11ea-952d-dc3a248e7bcc.png)

Image segmented into 200 superpixels

![Screenshot from 2020-04-24 23-49-42](https://user-images.githubusercontent.com/41234408/80244599-d83f7b00-8686-11ea-8a33-728790f45109.png)

Image segmented into 300 superpixels

![Screenshot from 2020-04-24 23-49-39](https://user-images.githubusercontent.com/41234408/80244625-e2617980-8686-11ea-9830-8693839bfb5f.png)

# 23) Colors in superpixels

The code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/colored_regions.py).

This helps in detecting the saturation of colors in different parts of an image that have been segmented into superpixels.

![Screenshot from 2020-04-26 13-27-49](https://user-images.githubusercontent.com/41234408/80301559-7ab04900-87c2-11ea-9a65-3528779f2be8.png)

# 24) Image Inpainting using OpenCV

The code for this can be found [here](https://github.com/yashk2000/Image-Processing/blob/master/opencv_inpainting.py).

Image inpainting is used for the removal of unwanted objects from an image or in restoring damaged portions of old photos.
We need to provide the input image with the defects and a mask which determines the regions to be *inpainted*.

![Screenshot from 2020-05-20 23-56-18](https://user-images.githubusercontent.com/41234408/82483253-bacec700-9af5-11ea-9691-d73e56f28d93.png)