Android Opengl 圆形纹理

Android Opengl Circle Texture

我一直在尝试添加纹理,但纹理如下所示:

这是我添加坐标和添加纹理坐标的代码。

for(int i=0; i<outerVertexCount; i++) {
    float percent=(i/(float)(outerVertexCount));
    float rad= (float) (percent*2*Math.PI);
    float outerX= (float) (centerX+radius*Math.cos(rad));
    float outerY=(float)(centerY+radius*Math.sin(rad));

    int previous=idx;
    circleCoordinates[idx++]=outerX;
    circleCoordinates[idx++]=outerY;
    circleCoordinates[idx++]=(circleCoordinates[previous])/(radius*2);
    circleCoordinates[idx++] = (circleCoordinates[previous+1])/(radius*2);
}

请帮帮我。

您正在绘制一个圆并使用'normalized' 几乎相同的坐标来获取纹理坐标。所以你需要做的是将纹理坐标归一化为纹理space[0, 1]。最简单的方法是检查顶部和左侧的纹理坐标是否应分别为 (0.5, 0.0) 和 (0.0, 0.5)。底部和右侧也应分别为 (0.5, 1.0) 和 (1.0, 0.5)。 (接触边缘的点)。

您的代码中的问题是:

  circleCoordinates[idx++] = (circleCoordinates[previous])/(radius*2);
  circleCoordinates[idx++] = (circleCoordinates[previous+1])/(radius*2);

当您在圆圈中创建点时,这些点将低于 0,在某些象限中会采用错误的纹理点。 您还可以通过 centerX and centerY 偏移您的圆圈,这也会移动您的纹理。

工作代码应该是:

  circleCoordinates[idx++] = (circleCoordinates[previous])/(radius*2) + 0.5 -  centerX / (radius * 2);
  circleCoordinates[idx++] = (circleCoordinates[previous+1])/(radius*2) + 0.5 - centerY / (radius * 2);

0.5 是因为您使用 radius * 2 对其进行了归一化,而没有添加 centerPoint 它将映射到 [-0.5, 0.5] 但是您还添加了中心点并将其归一化,因此您必须减去 center(XY) / (radius * 2) 移除神器。

TLDR;您的 (0,0) 是您希望它在左上角的第一张图像中的白色交点。相应地规范化。

这是一个例子:

public class Circle {

private int[] textureIDs = new int[1];

private int numberOfVertices = 30;


private final float[][] vertices = new float[numberOfVertices][2];
private final float[][] texVertices = new float[numberOfVertices][2];


private FloatBuffer vertexBuffer; // Buffer for vertex-array
private FloatBuffer texBuffer;    // Buffer for texture-coords-array (NEW)


public Circle() {
    setUpVertices(1.0f);
    // Setup vertex-array buffer. Vertices in float. A float has 4 bytes
    ByteBuffer vbb = ByteBuffer.allocateDirect(vertices.length * 4 * 2);
    vbb.order(ByteOrder.nativeOrder()); // Use native byte order
    vertexBuffer = vbb.asFloatBuffer(); // Convert byte buffer to float

    // Loop through the vertices and put them in the vertexbuffer
    for (int i = 0; i < numberOfVertices; i++) {
        for (int j = 0; j <= 1; j++) {
            vertexBuffer.put(vertices[i][j]); // Copy data into buffer
        }
    }
    vertexBuffer.position(0); // Rewind

    setUpTextureVertices(1.0f);

    // Setup texture-coords-array buffer, in float. An float has 4 bytes
    ByteBuffer tbb = ByteBuffer.allocateDirect(vertices.length * 4 * 2);
    tbb.order(ByteOrder.nativeOrder());
    texBuffer = tbb.asFloatBuffer();
    // Loop through the vertices and put them in the vertexbuffer
    for (int i = 0; i < numberOfVertices; i++) {
        for (int j = 0; j <= 1; j++) {
            texBuffer.put(texVertices[i][j]); // Copy data into buffer
        }
    }
    texBuffer.position(0);
}


public void draw(GL10 gl) {
    gl.glFrontFace(GL10.GL_CCW);    // Front face in counter-clockwise orientation
    gl.glEnable(GL10.GL_CULL_FACE); // Enable cull face
    gl.glCullFace(GL10.GL_BACK);    // Cull the back face (don't display)

    // Enable vertex-array and define its buffer
    gl.glEnableClientState(GL10.GL_VERTEX_ARRAY);
    gl.glEnableClientState(GL10.GL_TEXTURE_COORD_ARRAY); // Enable texture-coords-array
    gl.glVertexPointer(2, GL10.GL_FLOAT, 0, vertexBuffer);
    gl.glTexCoordPointer(2, GL10.GL_FLOAT, 0, texBuffer); // Define texture-coords


    // Draw the primitives from the vertex-array directly

    gl.glPushMatrix();
    gl.glTranslatef(0.0f, 0.0f, 1.0f);
    gl.glDrawArrays(GL10.GL_TRIANGLE_FAN, 0, numberOfVertices);
    gl.glPopMatrix();

    gl.glDisableClientState(GL10.GL_TEXTURE_COORD_ARRAY);
    gl.glDisableClientState(GL10.GL_VERTEX_ARRAY);
    gl.glDisable(GL10.GL_CULL_FACE);

}


public void loadTexture(GL10 gl, Context context) {
    gl.glGenTextures(1, textureIDs, 0); // Generate texture-ID array
    gl.glBindTexture(GL10.GL_TEXTURE_2D, textureIDs[0]); // Bind to texture ID
    // Set up texture filters
    gl.glTexParameterf(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MIN_FILTER, GL10.GL_NEAREST);
    gl.glTexParameterf(GL10.GL_TEXTURE_2D, GL10.GL_TEXTURE_MAG_FILTER, GL10.GL_LINEAR);

    // Construct an input stream to texture image
    @SuppressLint("ResourceType")
    InputStream istream = context.getResources().openRawResource(R.drawable.circle);
    Bitmap bitmap;
    try {
        // Read and decode input as bitmap
        bitmap = BitmapFactory.decodeStream(istream);
    } finally {
        try {
            istream.close();
        } catch (IOException e) {
        }
    }

    // Build Texture from loaded bitmap for the currently-bind texture ID
    GLUtils.texImage2D(GL10.GL_TEXTURE_2D, 0, bitmap, 0);
    bitmap.recycle();
}

private void setUpTextureVertices(float radius) {
    float theta0 = (float) (2 * Math.PI / (numberOfVertices - 1));

    for (int i = 0; i < numberOfVertices; i++) {

        float theta = i * theta0;

        float c = (float) Math.cos(theta);
        float s = (float) Math.sin(theta);

        float x = radius * c;
        float y = radius * s;

        texVertices[i][0] = (x * 0.5f) + 0.5f;
        texVertices[i][1] = (y * 0.5f) + 0.5f;

    }
}

private void setUpVertices(float radius) {
    float theta0 = (float) (2 * Math.PI / (numberOfVertices - 1));

    for (int i = 0; i < numberOfVertices; i++) {

        float theta = i * theta0;

        float c = (float) Math.cos(theta);
        float s = (float) Math.sin(theta);

        float x = radius * c;
        float y = radius * s;

        vertices[i][0] = x;
        vertices[i][1] = y;

    }
}
}