Ask Your Question
0

Barrel distorsion for Google cardboard?

asked 2016-02-04 07:05:15 -0600

Yoga gravatar image

updated 2016-02-04 07:08:21 -0600

Dear all,

I have a Fuji W3 stereo cam. I can get two images out of it. One for left eye and one for right eye. Now I want to watch the stereo image on my Google cardboard with my Galaxy S4 or Galaxy Note 4. Is there any way using OpenCV to give a barrel distortion to both images to see my stereo image correctly through the Google cardboard viewer?

If you look here:

http://krpano.com/docu/images/krpano1...

you can see that the left and right eye image has a barrel distortion applied, if you watch this image with Google cardboard it appears correctly in your eyes. And this is what I need to do with my own stereo images as well.

So again can I use OpenCV to apply this barrel distortion to my image? If yes - how?

Thanks a lot!

edit retag flag offensive close merge delete

Comments

have a look here. I also think that I have another version of it that it can be used. I will check about it when I return home. However, there is still something that I couldn't understand and it lacks an answer, just in case you are interested to spend some time.

theodore gravatar imagetheodore ( 2016-02-04 07:54:03 -0600 )edit

1 answer

Sort by » oldest newest most voted
3

answered 2016-02-04 16:47:49 -0600

theodore gravatar image

updated 2016-02-05 04:27:54 -0600

Here is another way to achieve the same effect. Though it lacks the scale part. I was trying to understand that part and implemented here as well by asking the question in the link in my comment. Anyway, here is the code:

static void meshgrid(InputArray _xgv, InputArray _ygv, OutputArray _X, OutputArray _Y)
{
    Mat xgv = _xgv.getMat();
    Mat ygv = _ygv.getMat();

  cv::repeat(xgv.reshape(1,1), ygv.total(), 1, _X);
  cv::repeat(ygv.reshape(1,1).t(), 1, xgv.total(), _Y);
}

// helper function (maybe that goes somehow easier)
static void meshgrid/*Test*/(const cv::Range &xgv, const cv::Range &ygv, OutputArray _X, OutputArray _Y)
{
  std::vector<float> t_x, t_y;
  for (int i = xgv.start; i <= xgv.end; i++) t_x.push_back(i);
  for (int i = ygv.start; i <= ygv.end; i++) t_y.push_back(i);

  Mat __X, __Y;
  meshgrid(cv::Mat(t_x), cv::Mat(t_y), __X, __Y);

  Mat(__X).copyTo(_X);
  Mat(__Y).copyTo(_Y);
}

void radialDistortion(InputArray _src, OutputArray _dst, double Cx, double Cy, double coef, bool scale = true)
{
    // die if distortion parameters are not correct
    CV_Assert(Cx >= 0 && Cy >= 0/* && k >= 0*/);

    Mat src = _src.getMat();

    int ncols = src.cols;
    int nrows = src.rows;

    Mat xi, yi;
    meshgrid(Range(1, ncols), Range(1, nrows), xi, yi);

    int imid_x = Cx;
    int imid_y = Cy;

    Mat xt = xi - imid_x;
    Mat yt = yi - imid_y;

    xt = xt.t();
    yt = yt.t();

    if (!xt.isContinuous())
        xt = xt.clone();

    if (!yt.isContinuous())
        yt = yt.clone();

    xt = xt.reshape(0, xt.rows * xt.cols);
    yt = yt.reshape(0, yt.rows * yt.cols);

    Mat r, theta;
    cartToPolar(xt, yt, r, theta);

    Mat s = r + coef * r.mul(r.mul(r));

    Mat ut, vt;
    polarToCart(s, theta, ut, vt);

    Mat u, v;

    if (!ut.isContinuous())
        ut = ut.clone();

    if (!vt.isContinuous())
        vt = vt.clone();

    u = ut.reshape(0, ncols) + imid_x;
    v = vt.reshape(0, ncols) + imid_y;
    u = u.t();
    v = v.t();

    remap(src, _dst, u, v, CV_INTER_LINEAR, BORDER_CONSTANT);
}

int main()
{
    Mat input = imread("lena.jpg"/*, CV_LOAD_IMAGE_GRAYSCALE*/); // it works with grayscale images as well ;-)

    if(!input.data || input.empty())
        cout << "Problem loading image!!!" << endl;

    Mat output;
    radialDistortion(input, output, round(static_cast<float>(input.cols) / 2), round(static_cast<float>(input.rows) / 2), 0.000003, false);

    Mat canvas = Mat::zeros(input.rows, input.cols*2+10, input.type());

    input.copyTo(canvas(Range::all(), Range(0, output.cols)));
    output.copyTo(canvas(Range::all(), Range(output.cols+10, output.cols*2+10)));

    // if it is too big to fit on the screen, then scale it down by 2, hopefully it'll fit :-)
    if(canvas.cols > 1920)
    {
        resize(canvas, canvas, Size(canvas.cols/2, canvas.rows/2));
    }

    imshow("canvas", canvas);

    waitKey(0);
    return 0;
}

image description

if you have some time you can search more about it ;-).

edit flag offensive delete link more

Comments

Hi theodore,

sorry for beeing so lately, I was on vacation...

Thanks so much for the code snippet!! I will take a look at it!

Yoga gravatar imageYoga ( 2016-02-24 09:06:40 -0600 )edit

Question Tools

2 followers

Stats

Asked: 2016-02-04 07:05:15 -0600

Seen: 979 times

Last updated: Feb 05 '16