سؤال

I've drawn a grid at z=0 in OpenGL 2.0 ES, and just want to convert touch inputs to x/y coordinates on that plane. It seems like this is best done through ray tracing, which involves running gluUnProject on 0 and 1, then creating a ray, solving that ray for z=0?

I found this code, but it is OpenGL ES 1.0: i-schuetz / Android_OpenGL_Picking

Screenshot of app running so you can see camera distortion.

My code on Github, only 4 files. The unproject function I'm trying to write is in MyGLRenderer.java:

    public float[] unproject(float rx, float ry) {
            float rz = 1;
        float[] xyzw = {0, 0, 0, 0};
        int[] viewport = {0, 0, mDisplayWidth, mDisplayHeight};
        android.opengl.GLU.gluUnProject(
                        rx, ry, rz, // window coordinates
                        mViewMatrix, 0,
                        mProjectionMatrix, 0, 
                        viewport, 0,
                        xyzw, 0);
        xyzw[0] /= xyzw[3];
        xyzw[1] /= xyzw[3];
        xyzw[2] /= xyzw[3];
        xyzw[3] = 1;
        return xyzw;
    }

I would like this function to take an rx and ry for the screen, and return an rx and ry for the z=0 plane.

هل كانت مفيدة؟

المحلول

There is nothing particularly special about what gluUnProject (...) does. If you have all of the matrices and the viewport dimensions (x,y and width,height) I can walk you through the process of implementing it yourself.

NOTE: I tend to call each coordinate space by a different name than you might be used to, understand that screen space is another name for window space, view space is another name for eye space, object space is another name for model space.


Step 1:

   Screen Space to NDC space (Undo Viewport Transform)

          NDCX = (2.0 × (ScreenX - ViewportX) / ViewportW) - 1.0

          NDCY = (2.0 × (ScreenY - ViewportY) / ViewportH)  - 1.0

   Screen Space to NDC space (Undo Depth Range Mapping)

         Typically in screen space, the Depth Range will map z=0 to near and z=1 to far:

               NDCZ = 2.0 × ScreenZ - 1.0


Step 2:

   NDC space to Object space (Undo Projection, View and Model Transforms)

          (Projection Matrix)-1  × NDCXYZ1  = ViewXYZW

          (ModelView Matrix)-1 × ViewXYZW = ObjectXYZW

                    This can actually be combined into a single step, as you will see below...


                           ObjectXYZw = (Projection Matrix × ModelView Matrix)-1 × NDCXYZ1


Now, you may notice that I crossed-out W in ObjectXYZ, we really do not care about this at all but the math will produce a pesky W value nevertheless. At this point, you can return the individual components of ObjectXYZ as your rX, rY and rZ.

مرخصة بموجب: CC-BY-SA مع الإسناد
لا تنتمي إلى StackOverflow
scroll top