0

我已经实现了一种颜色选择方法,它也有效;有时。问题是当我在方法中调用我的onSurfaceChanged方法时,它确实读回了正确的像素。但是当我在 GLSurfaceView 中调用我的方法时onTouchevent,它只读回零。在我的方法中,我创建了一个framebuffer object并附加了两个renderbuffer objects。我的渲染器设置为RENDERMODE_WHEN_DIRTYglGetError()返回 0。问题到底出在哪里?我不确定这是否是我的方法,或者我是否根本无法在onTouchevent. 这是我在渲染器中的选取方法的源代码:

public int pick(int x, int y) {
    int result = -2;

    int[]  view = new int[4];
    GLES20.glGetIntegerv(GLES20.GL_VIEWPORT, view, 0);

    y = view[3] - y;

    int[] fbo = new int[1];
    GLES20.glGenFramebuffers(1, fbo, 0);
    GLES20.glBindFramebuffer(GLES20.GL_FRAMEBUFFER, fbo[0]);

    int[] rbo = new int[2];
    GLES20.glGenRenderbuffers(2, rbo, 0);

    GLES20.glBindRenderbuffer(GLES20.GL_RENDERBUFFER, rbo[0]);
    GLES20.glRenderbufferStorage(
            GLES20.GL_RENDERBUFFER, 
            GLES20.GL_RGBA4, view[2], view[3]);

    GLES20.glFramebufferRenderbuffer(
            GLES20.GL_FRAMEBUFFER, GLES20.GL_COLOR_ATTACHMENT0, 
            GLES20.GL_RENDERBUFFER, rbo[0]);

    GLES20.glBindRenderbuffer(GLES20.GL_RENDERBUFFER, rbo[1]);
    GLES20.glRenderbufferStorage(
            GLES20.GL_RENDERBUFFER, 
            GLES20.GL_DEPTH_COMPONENT16, 
            view[2], view[3]);

    GLES20.glFramebufferRenderbuffer(
            GLES20.GL_FRAMEBUFFER, GLES20.GL_DEPTH_ATTACHMENT, 
            GLES20.GL_RENDERBUFFER, rbo[1]);

    int status = GLES20.glCheckFramebufferStatus(GLES20.GL_FRAMEBUFFER);
    if (status == GLES20.GL_FRAMEBUFFER_COMPLETE) {
        result = -1;
        GLES20.glClear(
                GLES20.GL_COLOR_BUFFER_BIT | 
                GLES20.GL_DEPTH_BUFFER_BIT);

        for (int i = 0; i < mObjects.size(); i++)
            mObjects.get(i).render(mProgram, mMatrixMVP, true);

        ByteBuffer pixels = ByteBuffer.allocate(view[2] * view[3] * 4);
        pixels.order(ByteOrder.nativeOrder());

        GLES20.glReadPixels( // I read every pixel just for debugging
                0, 0, view[2], view[3], GLES20.GL_RGBA, 
                GLES20.GL_UNSIGNED_BYTE, pixels);

        int e = GLES20.glGetError(); // always returns 0
        byte[] tmp = pixels.array(); // only zeros when called in onTouch

        for (int i = 0; i < mObjects.size(); i++)
            if ((Math.abs(r - mObjects.get(i).CODE_R) < 8) &&
                (Math.abs(g - mObjects.get(i).CODE_G) < 8) &&
                (Math.abs(b - mObjects.get(i).CODE_B) < 8)) {

                result = i;
                break;
            }
    } 

    GLES20.glDeleteRenderbuffers(2, rbo, 0);
    GLES20.glDeleteFramebuffers(1, fbo, 0);
    GLES20.glBindFramebuffer(GLES20.GL_FRAMEBUFFER, 0);

    return result;
}

由于它可能是相关的,这是我的onDrawFrame方法:

@Override
public void onDrawFrame(GL10 gl) {
    GLES20.glClear(
            GLES20.GL_COLOR_BUFFER_BIT | 
            GLES20.GL_DEPTH_BUFFER_BIT);

    for (int i = 0; i < mObjects.size(); i++)
        mObjects.get(i).render(mProgram, mMatrixMVP, false);
}

我读过 onDrawFrame 方法在单独的线程中运行,但我不认为这是问题所在,因为我在RENDERMODE_WHEN_DIRTY...

4

2 回答 2

0

即使您使用RENDERMODE_WHEN_DIRTY渲染器在不同的线程上运行,它也不会渲染,直到您调用requestRender(). 您可以尝试使用queueEvent()-method 在渲染线程上运行代码,看看是否有帮助。您还可以更改您的pick()-method 以便它在内部使用 aRunnablequeueEvent-method(或 a Handler)。您当然还需要使用某种回调才能“返回”结果。

于 2013-06-20T10:44:13.420 回答
0

你为什么不检查这个库。我认为它非常好,您可以使用它来代替创建新的颜色选择器。

https://github.com/LarsWerkman/HoloColorPicker

于 2013-06-20T10:09:37.363 回答