Since I found the above challenge quite interesting I worked a bit on it. My approach is based mainly on morphological operations, and without applying any complex technique the result seems pretty much nice. Anyway, here is my approach:
#include <iostream>
#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;
int main()
{
// Load source image
string filename = "notes.png";
Mat src = imread(filename);
// Check if image is loaded fine
if(!src.data)
cerr << "Problem loading image!!!" << endl;
// Show source image
imshow("src", src);
data:image/s3,"s3://crabby-images/35388/3538843e15071a22a7a7222a661549d463cc307f" alt="image description"
// Transform source image to gray it is not
Mat gray;
if (src.channels() == 3)
{
cvtColor(src, gray, CV_BGR2GRAY);
}
else
{
gray = src;
}
// Show gray image
imshow("gray", gray);
data:image/s3,"s3://crabby-images/b69c6/b69c62498ac35949b8e0fb8dd5c38a9ee2e39239" alt="image description"
// Apply adaptiveThreshold at the bitwise_not of gray, notice the ~ symbol
Mat bw;
adaptiveThreshold(~gray, bw, 255, CV_ADAPTIVE_THRESH_MEAN_C, THRESH_BINARY, 15, -2);
// Show binary image
imshow("binary", bw);
data:image/s3,"s3://crabby-images/5fe15/5fe1527745f652ae595c6ecb60aa86d818fd63c5" alt="image description"
// Create the images that will use to extract the horizonta and vertical lines
Mat horizontal = bw.clone();
Mat vertical = bw.clone();
// Specify size on horizontal axis
int horizontalsize = horizontal.cols / 30;
// Create structure element for extracting horizontal lines through morphology operations
Mat horizontalStructure = getStructuringElement(MORPH_RECT, Size(horizontalsize,1));
// Apply morphology operations
erode(horizontal, horizontal, horizontalStructure, Point(-1, -1));
dilate(horizontal, horizontal, horizontalStructure, Point(-1, -1));
// Show extracted horizontal lines
imshow("horizontal", horizontal);
data:image/s3,"s3://crabby-images/5a2bb/5a2bbe47916f477a186b959eb42e8f5b3fec8d5e" alt="image description"
// Specify size on vertical axis
int verticalsize = vertical.rows / 30;
// Create structure element for extracting vertical lines through morphology operations
Mat verticalStructure = getStructuringElement(MORPH_RECT, Size( 1,verticalsize));
// Apply morphology operations
erode(vertical, vertical, verticalStructure, Point(-1, -1));
dilate(vertical, vertical, verticalStructure, Point(-1, -1));
// Show extracted vertical lines
imshow("vertical", vertical);
data:image/s3,"s3://crabby-images/c425f/c425ffbf7a1bf04bb30ffe67b049c5a98d347101" alt="image description"
// Inverse vertical image
bitwise_not(vertical, vertical);
imshow("vertical_bit", vertical);
data:image/s3,"s3://crabby-images/a7c42/a7c4220bca1cd9f17ce98278cb25971ea0c62a81" alt="image description"
// Extract edges and smooth image according to the logic
// 1. extract edges
// 2. dilate(edges)
// 3. src.copyTo(smooth)
// 4. blur smooth img
// 5. smooth.copyTo(src, edges)
// Step 1
Mat edges;
adaptiveThreshold(vertical, edges, 255, CV_ADAPTIVE_THRESH_MEAN_C, THRESH_BINARY, 3, -2);
imshow("edges", edges);
data:image/s3,"s3://crabby-images/1a363/1a363b24be4e6f33c98c67cffe1d2b1957b38295" alt="image description"
// Step 2
Mat kernel = Mat::ones(2, 2, CV_8UC1);
dilate(edges, edges, kernel);
imshow("dilate", edges);
data:image/s3,"s3://crabby-images/4d72b/4d72bc4508c5ecc38043c9fd940c92c6a06802ce" alt="image description"
// Step 3
Mat smooth;
vertical.copyTo(smooth);
// Step 4
blur(smooth, smooth, Size(2, 2));
// Step 5
smooth.copyTo(vertical, edges);
// Show final result
imshow("smooth", vertical);
data:image/s3,"s3://crabby-images/a7c17/a7c1764c6325750167c2ef46f1e5bd113513b399" alt="image description"
waitKey(0);
return 0;
}
enjoy.
Start by adding an example image please.
I've added 2 images. Do you know the algorithm to solve it ? @StevenPuttemans