[Reference Site]
Image 1 | Image 2 |
Using OpenCV API transformation (image 2 → image 1) | Not using OpenCV API transformation (image 2 → image 1) |
cv::findHomography
API 적용Basic Code
std::vector<cv::Point2f> img1_2d, img2_2d;
cv::Mat H = cv::findHomography(img1_2d, img2_2d, CV_RANSAC, 1);
cv::Mat warpImg = calcROIImg(img2, H, x_diff, y_diff, x_min, y_min, x_max, y_max);
// WarpImg pixels
for(int y = 0; y < img2.rows; y++)
{
for(int x = 0; x < img2.cols; x++)
{
// Get RGB values
cv::Vec3b rgb = img2.at<cv::Vec3b>(y, x);
cv::Mat tmp_point(3,1,cv::DataType<double>::type);
tmp_point.at<double>(0,0) = x;
tmp_point.at<double>(1,0) = y;
tmp_point.at<double>(2,0) = 1;
cv::Mat H_tmp_point = H.inv() * tmp_point;
H_tmp_point = H_tmp_point / H_tmp_point.at<double>(2,0);
H_tmp_point.at<double>(0,0) = H_tmp_point.at<double>(0,0) - x_min;
H_tmp_point.at<double>(1,0) = H_tmp_point.at<double>(1,0) - y_min;
warpImg.at<cv::Vec3b>(H_tmp_point.at<double>(1,0), H_tmp_point.at<double>(0,0)) = rgb;
}
}
Code Results
Basic Linear Interpolation Code
cv::Vec3b PixelLinearInter(cv::Vec3b pt1, cv::Vec3b pt2, double t)
{
int interpolation_r = (int)pt1[0] + t * ((int)pt2[0] - (int)pt1[0]);
int interpolation_g = (int)pt1[1] + t * ((int)pt2[1] - (int)pt1[1]);
int interpolation_b = (int)pt1[2] + t * ((int)pt2[2] - (int)pt1[2]);
cv::Vec3b inter_rgb(interpolation_r, interpolation_g, interpolation_b);
return inter_rgb;
}
Basic Code
// Linear Interpolation pixels
cv::Mat interImg = warpImg.clone();
for(int y = 1; y < warpImg.rows - 1; y++)
{
for(int x = 1; x < warpImg.cols - 1; x++)
{
int cur_r = (int)warpImg.at<cv::Vec3b>(y, x)[0];
int cur_g = (int)warpImg.at<cv::Vec3b>(y, x)[1];
int cur_b = (int)warpImg.at<cv::Vec3b>(y, x)[2];
int count = 0;
for(int p = -1; p <= 1; p++)
{
for(int q = -1; q <= 1; q++)
{
if(p == 0 && q == 0)
continue;
int tmp_r = (int)warpImg.at<cv::Vec3b>(y+p, x+q)[0];
int tmp_g = (int)warpImg.at<cv::Vec3b>(y+p, x+q)[1];
int tmp_b = (int)warpImg.at<cv::Vec3b>(y+p, x+q)[2];
if(tmp_r == 0 && tmp_g == 0 && tmp_b == 0)
count++;
}
}
if(cur_r == 0 && cur_g == 0 && cur_b == 0)
{
if(count >= 0 && count < 7)
{
bool minus = false;
bool plus = false;
int final_inter_r, final_inter_g, final_inter_b;
cv::Vec3b pt1 = warpImg.at<cv::Vec3b>(y-1, x-1);
cv::Vec3b pt2 = warpImg.at<cv::Vec3b>(y+1, x+1);
cv::Vec3b inter_pt1 = PixelLinearInter(pt1, pt2, 0.5);
cv::Vec3b pt3 = warpImg.at<cv::Vec3b>(y-1, x+1);
cv::Vec3b pt4 = warpImg.at<cv::Vec3b>(y+1, x-1);
cv::Vec3b inter_pt2 = PixelLinearInter(pt3, pt4, 0.5);
cv::Vec3b compare(0, 0, 0);
if(pt1 == compare || pt2 == compare)
{
final_inter_r = (int)inter_pt2[0];
final_inter_g = (int)inter_pt2[1];
final_inter_b = (int)inter_pt2[2];
}
else if(pt3 == compare || pt4 == compare)
{
final_inter_r = (int)inter_pt1[0];
final_inter_g = (int)inter_pt1[1];
final_inter_b = (int)inter_pt1[2];
}
else
{
final_inter_r = ((int)inter_pt1[0] + (int)inter_pt2[0]) / 2;
final_inter_g = ((int)inter_pt1[1] + (int)inter_pt2[1]) / 2;
final_inter_b = ((int)inter_pt1[2] + (int)inter_pt2[2]) / 2;
}
cv::Vec3b final_inter_pt(final_inter_r, final_inter_g, final_inter_b);
interImg.at<cv::Vec3b>(y, x) = final_inter_pt;
}
}
}
}
좌상-우하 (기울기가 마이너스) 경우에 대한 Linear interpolation 결과 | 좌하-우상 (기울기가 플러스) 경우에 대한 Linear interpolation 결과 |
Basic Code
cv::Mat AddHomographyImg(cv::Mat img1, cv::Mat img2, cv::Mat warp_img2, cv::Mat H)
{
double x_diff, y_diff;
double x_min, y_min, x_max, y_max;
cv::Mat ROIImg = calcROIImg(img2, H, x_diff, y_diff, x_min, y_min, x_max, y_max);
// Translate img1 for warp img2
cv::Mat img1_trans = img1.clone();
if(x_min <= 0 && y_min > 0)
{
cv::Mat x_trans_img1(img1.rows, (img1.cols - x_min), img1.type(), cv::Scalar(0,0,0));
for(int y = 0; y < img1.rows; y++)
{
for(int x = 0; x < img1.cols; x++)
{
cv::Vec3b rgb = img1.at<cv::Vec3b>(y, x);
x_trans_img1.at<cv::Vec3b>(y, x - x_min) = rgb;
}
}
img1_trans = x_trans_img1.clone();
}
else if(x_min > 0 && y_min < 0)
{
cv::Mat y_trans_img1((img1.rows - y_min), img1.cols, img1.type(), cv::Scalar(0,0,0));
for(int y = 0; y < img1.rows; y++)
{
for(int x = 0; x < img1.cols; x++)
{
cv::Vec3b rgb = img1.at<cv::Vec3b>(y, x);
y_trans_img1.at<cv::Vec3b>(y - y_min, x) = rgb;
}
}
img1_trans = y_trans_img1.clone();
}
else if(x_min < 0 && y_min < 0)
{
cv::Mat xy_trans_img1((img1.rows - y_min), (img1.cols - x_min), img1.type(), cv::Scalar(0,0,0));
for(int y = 0; y < img1.rows; y++)
{
for(int x = 0; x < img1.cols; x++)
{
cv::Vec3b rgb = img1.at<cv::Vec3b>(y, x);
xy_trans_img1.at<cv::Vec3b>(y - y_min, x - x_min) = rgb;
}
}
img1_trans = xy_trans_img1.clone();
}
double img1_col = img1_trans.cols;
double img1_row = img1_trans.rows;
double img2_col = warp_img2.cols;
double img2_row = warp_img2.rows;
cv::Mat panoImg;
if(img1_col <= img2_col && img1_row <= img2_row)
{
cv::Mat transImg(img2_row, img2_col, img1.type(), cv::Scalar(0,0,0));
panoImg = transImg.clone();
}
else if(img1_col <= img2_col && img1_row > img2_row)
{
cv::Mat transImg(img1_row, img2_col, img1.type(), cv::Scalar(0,0,0));
panoImg = transImg.clone();
}
else if(img1_col > img2_col && img1_row <= img2_row)
{
cv::Mat transImg(img2_row, img1_col, img1.type(), cv::Scalar(0,0,0));
panoImg = transImg.clone();
}
else
{
cv::Mat transImg(img1_row, img1_col, img1.type(), cv::Scalar(0,0,0));
panoImg = transImg.clone();
}
std::cout << "trans img size: " << img1_trans.size() << std::endl;
std::cout << "pano img size: " << panoImg.size() << std::endl;
// Add two imgs
for(int y = 0; y < panoImg.rows; y++)
{
for(int x = 0; x < panoImg.cols; x++)
{
int final_r, final_g, final_b;
cv::Vec3b compare(0, 0, 0);
if(img1_col <= x && x < img2_col)
{
cv::Vec3b cur_rgb2 = warp_img2.at<cv::Vec3b>(y, x);
if(cur_rgb2 == compare)
continue;
else
{
final_r = (int)cur_rgb2[0];
final_g = (int)cur_rgb2[1];
final_b = (int)cur_rgb2[2];
}
cv::Vec3b final_pt(final_r, final_g, final_b);
panoImg.at<cv::Vec3b>(y, x) = final_pt;
continue;
}
else if(img2_col <= x && x < img1_col)
{
cv::Vec3b cur_rgb1 = img1_trans.at<cv::Vec3b>(y, x);
if(cur_rgb1 == compare)
continue;
else
{
final_r = (int)cur_rgb1[0];
final_g = (int)cur_rgb1[1];
final_b = (int)cur_rgb1[2];
}
cv::Vec3b final_pt(final_r, final_g, final_b);
panoImg.at<cv::Vec3b>(y, x) = final_pt;
continue;
}
if(img1_row <= y && y < img2_row)
{
cv::Vec3b cur_rgb2 = warp_img2.at<cv::Vec3b>(y, x);
if(cur_rgb2 == compare)
continue;
else
{
final_r = (int)cur_rgb2[0];
final_g = (int)cur_rgb2[1];
final_b = (int)cur_rgb2[2];
}
cv::Vec3b final_pt(final_r, final_g, final_b);
panoImg.at<cv::Vec3b>(y, x) = final_pt;
continue;
}
else if(img2_row <= y && y < img1_row)
{
cv::Vec3b cur_rgb1 = img1_trans.at<cv::Vec3b>(y, x);
if(cur_rgb1 == compare)
continue;
else
{
final_r = (int)cur_rgb1[0];
final_g = (int)cur_rgb1[1];
final_b = (int)cur_rgb1[2];
}
cv::Vec3b final_pt(final_r, final_g, final_b);
panoImg.at<cv::Vec3b>(y, x) = final_pt;
continue;
}
// Get RGB values
cv::Vec3b cur_rgb1 = img1_trans.at<cv::Vec3b>(y, x);
cv::Vec3b cur_rgb2 = warp_img2.at<cv::Vec3b>(y, x);
if(cur_rgb1 == compare && cur_rgb2 == compare)
continue;
else if(cur_rgb1 == compare && cur_rgb2 != compare)
{
final_r = (int)cur_rgb2[0];
final_g = (int)cur_rgb2[1];
final_b = (int)cur_rgb2[2];
}
else if(cur_rgb1 != compare && cur_rgb2 == compare)
{
final_r = (int)cur_rgb1[0];
final_g = (int)cur_rgb1[1];
final_b = (int)cur_rgb1[2];
}
else
{
final_r = ((int)cur_rgb1[0] + (int)cur_rgb2[0]) / 2;
final_g = ((int)cur_rgb1[1] + (int)cur_rgb2[1]) / 2;
final_b = ((int)cur_rgb1[2] + (int)cur_rgb2[2]) / 2;
}
cv::Vec3b final_pt(final_r, final_g, final_b);
panoImg.at<cv::Vec3b>(y, x) = final_pt;
}
}
return panoImg;
}
Code Results
OpenCV 기반 warp API 사용 후 direct stitching 한 결과 | Homography 추정 후 OpenCV warp API 사용하지 않고 direct stitching 한 결과 |
[Stitching Time]
Method | Auto Stitching | Fast Stitching API | Only Homography Stitching | Only Homography Stitching (not use warp API) |
---|---|---|---|---|
Time (sec) | 4.00 | 0.9 | 0.02 | 0.37 |