2011-12-31 21 views
7

Estoy tratando de dibujar texturas en quads en OpenGL2.0. Hasta ahora conseguí que aparezcan los cuádriceps y todo, pero las texturas no están allí, los cuádriceps son todos negros.Android OpenGL2.0 mostrando texturas negras

Mi sospecha principal es que no soy la cartografía de las texturas correctamente - mis texturas son no potencias de 2, ni son cuadrados - su anchura está en el campo de manchura y su altura en mRowHeight.

Los quads se dibujan en una lista vertical, que se realiza con una matriz de traducción.

Estaré muy agradecido si alguien puede pasar por esta razón, estoy desesperado.

Aquí es código relacionado:

inicialización de los tampones:

void initBuffers() { 
     float r = (float) mRowHeight/(mHeight/2f); 
     ByteBuffer bb; 

     float[] bounds = { // X Y Z 
     /* 0 - TL */-1f, 1f, 0f, 
     /* 1 - BL */-1f, 1 - r, 0f, 
     /* 2 - BR */ 1f, 1 - r, 0f, 
     /* 3 - TR */ 1f, 1f, 0f }; 
     bb = ByteBuffer.allocateDirect(bounds.length * 4); 
     bb.order(ByteOrder.nativeOrder()); 
     mVertBuffer = bb.asFloatBuffer(); 
     mVertBuffer.put(bounds).position(0); 

     short[] indices = { 0, 1, 2, 0, 2, 3 }; 
     bb = ByteBuffer.allocateDirect(indices.length * 2); 
     bb.order(ByteOrder.nativeOrder()); 
     mIndBuffer = bb.asShortBuffer(); 
     mIndBuffer.put(indices).position(0); 

     float[] texture = { 
       /* 0 - BL */-1f, 1f, 
       /* 1 - TL */-1f, 1 - r, 
       /* 2 - BR */1f, 1 - r, 
       /* 3 - TR */1f, 1f }; 
     bb = ByteBuffer.allocateDirect(texture.length * 4); 
     bb.order(ByteOrder.nativeOrder()); 
     mTexBuffer = bb.asFloatBuffer(); 
     mTexBuffer.put(texture).position(0); 

    } 

dibujo del marco:

@Override 
    public void drawFrame() { 
     long startTime = System.currentTimeMillis(); 
     if (!mLayoutComplete) { 
      return; 
     } 
     final float halfw = mHeight/2f; 
     final int rowHeight = mRowHeight; 
     final float r = (float) rowHeight/halfw; 

     int i = mFirstRow; 

     final float initial = (float) ((i * rowHeight) - mScroll)/halfw; 

     Matrix.setIdentityM(mTMatrix, 0); 
     Matrix.translateM(mTMatrix, 0, 0, -initial, 0); 

     GLES20.glVertexAttribPointer(maPositionHandle, 3, GLES20.GL_FLOAT, 
       false, 0, mVertBuffer); 
     GLES20.glEnableVertexAttribArray(maPositionHandle); 

     final int l = mLastRow; 
     for (; i <= l; i++) { 

      ThumbRow thr = mCache.get(i); 
      if (thr != null) { 
       GLES20.glActiveTexture(GLES20.GL_TEXTURE0); 
       if (thr.mDirty) { 
        thr.loadTexture(null, null); 
        thr.mDirty = false; 
       } else { 
        GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, thr.mRow + 1); 
       } 
       GLES20.glUniform1i(msTextureHandle, 0); 

       GLES20.glVertexAttribPointer(maTextureHandle, 2, 
         GLES20.GL_FLOAT, false, 0, mTexBuffer); 
       GLES20.glEnableVertexAttribArray(maTextureHandle); 

       GLES20.glUniformMatrix4fv(muTMatrixHandle, 1, false, 
         mTMatrix, 0); 
       GLES20.glDrawElements(GLES20.GL_TRIANGLES, 6, 
         GLES20.GL_UNSIGNED_SHORT, mIndBuffer); 
       Log.i(TAG, GLES20.glGetProgramInfoLog(mProgram)); 
      } 

      Matrix.translateM(mTMatrix, 0, 0, -r, 0); // Shift drawing 
                 // window to next 
                 // row. 
     } 

     GLES20.glDisableVertexAttribArray(maPositionHandle); 
     GLES20.glDisableVertexAttribArray(maTextureHandle); 

     Log.d(TAG, "onDrawFrame(): " 
       + (System.currentTimeMillis() - startTime)); 
    } 

Cargando las texturas:

 public void loadTexture(GL10 gl, Context c) { 
      GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, mRow + 1); 

      // Create Nearest Filtered Texture 
      GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, 
        GLES20.GL_TEXTURE_MIN_FILTER, GLES20.GL_NEAREST); 
      GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, 
        GLES20.GL_TEXTURE_MAG_FILTER, GLES20.GL_LINEAR); 

      // Different possible texture parameters, e.g. 
      // GLES20.GL_CLAMP_TO_EDGE 
      GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, 
        GLES20.GL_TEXTURE_WRAP_S, GLES20.GL_REPEAT); 
      GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, 
        GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_REPEAT); 

      Bitmap bitmap = mBitmap; 
      if (bitmap == null) { 
       bitmap = mEmptyBitmap; 
      } 
      GLUtils.texImage2D(GLES20.GL_TEXTURE_2D, 0, bitmap, 0); 
     } 

Vertex Shader:

uniform mat4 uTMatrix; 
uniform mat4 uMVPMatrix; 
attribute vec4 aPosition; 
attribute vec2 aTextureCoord; 
varying vec2 vTextureCoord; 

void main() { 
    gl_Position = uMVPMatrix * uTMatrix * aPosition; 
    vTextureCoord = aTextureCoord; 
} 

Fragmento de Shader:

precision mediump float; 
varying vec2 vTextureCoord; 
uniform sampler2D sTexture; 
void main() { 
    gl_FragColor = texture2D(sTexture, vTextureCoord); 
} 

será muy apreciada Cualquier ayuda.

+0

Una vez que se llama '' glBindTexture (..) '' ¿qué es '' mRow + 1'' por ejemplo? Si se trata de una variable interna, debe cambiarla por el nombre de textura que obtiene de '' glGenTextures (..) ''. Y, fundamentalmente, si solo está usando una textura (y solo actualiza su contenido), use ese nombre de textura en todo momento. – harism

+0

mRow es el índice del quad actual que se está dibujando: la textura del quad está determinada por mRow. EDITAR: por lo que entendí, está bien usar mis propios nombres de texturas, siempre que sean únicos y los recuerdo – saarraz1

+0

No. Debes asignar nombres de texturas usando glGenTextures (..), no puedes crear los propios. En su caso, debe asignar ROW_COUNT nombres de texturas y usar glBindTexture (.., TEXTURE_NAMES [mRowIdx]).Pero si ROW_COUNT es muy alto, puede ser mejor volver a cargar el mapa de bits en una textura solo para cada fila para evitar el uso innecesario de la memoria. – harism

Respuesta

20

Este es un problema bastante común de texturizado en OpenGL ES 2.0 que me causó muchos dolores de cabeza en el pasado.

En OpenGL ES 2.0, en caso de no poder de 2 texturas, el modo de ajuste solo puede ser GL_CLAMP_TO_EDGE.

Hay restricciones así en el filtro se puede utilizar que puede ser única GL_NEAREST o GL_LINEAR (en otras palabras, no Mipmapping)

En palabras sencillas, que la comprobación LoadTexture función, cambiar estas 2 líneas de código a partir de:

GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, 
        GLES20.GL_TEXTURE_WRAP_S, GLES20.GL_REPEAT); 
GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, 
        GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_REPEAT); 

a

GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, 
        GLES20.GL_TEXTURE_WRAP_S, GLES20.GL_CLAMP_TO_EDGE); 
GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, 
        GLES20.GL_TEXTURE_WRAP_T, GLES20.GL_CLAMP_TO_EDGE); 

espero que esto soluciona el problema, hágamelo saber :)

Maurizio Benedetti

+0

OMG IT WORKS !! No puedo agradecerte lo suficiente! Casi había renunciado al uso de OGL luego apareciste. Muchas gracias de nuevo. – saarraz1

+0

guau, feliz de leer eso. Que tengas un buen día y buena suerte para tus proyectos. –

+0

Tx. Me ayudó también. Supongo que es necesario leer las especificaciones :) –

Cuestiones relacionadas