package org.andengine.entity.primitive;
import org.andengine.engine.camera.Camera;
import org.andengine.entity.shape.IShape;
import org.andengine.entity.shape.RectangularShape;
import org.andengine.entity.shape.Shape;
import org.andengine.opengl.shader.PositionColorTextureCoordinatesShaderProgram;
import org.andengine.opengl.shader.constants.ShaderProgramConstants;
import org.andengine.opengl.texture.region.ITextureRegion;
import org.andengine.opengl.util.GLState;
import org.andengine.opengl.vbo.HighPerformanceVertexBufferObject;
import org.andengine.opengl.vbo.IVertexBufferObject;
import org.andengine.opengl.vbo.DrawType;
import org.andengine.opengl.vbo.VertexBufferObjectManager;
import org.andengine.opengl.vbo.attribute.VertexBufferObjectAttribute;
import org.andengine.opengl.vbo.attribute.VertexBufferObjectAttributes;
import org.andengine.opengl.vbo.attribute.VertexBufferObjectAttributesBuilder;
import org.andengine.util.debug.Debug;
import org.andengine.util.exception.MethodNotSupportedException;
import android.opengl.GLES20;
import android.renderscript.Mesh;
/**
* (c) Zynga 2012
*
* @author Nicolas Gramlich <ngramlich@zynga.com>
* @since 16:44:50 - 09.02.2012
*/
public class TexturedMesh extends Shape {
// ===========================================================
// Constants
// ===========================================================
public static final int VERTEX_INDEX_X = 0;
public static final int VERTEX_INDEX_Y = TexturedMesh.VERTEX_INDEX_X + 1;
public static final int COLOR_INDEX = TexturedMesh.VERTEX_INDEX_Y + 1;
public static final int TEXTURECOORDINATES_INDEX_U = TexturedMesh.COLOR_INDEX + 1;
public static final int TEXTURECOORDINATES_INDEX_V = TexturedMesh.TEXTURECOORDINATES_INDEX_U + 1;
public static final int VERTEX_SIZE = 5;
public static final VertexBufferObjectAttributes VERTEXBUFFEROBJECTATTRIBUTES_DEFAULT = new VertexBufferObjectAttributesBuilder(3)
.add(ShaderProgramConstants.ATTRIBUTE_POSITION_LOCATION, ShaderProgramConstants.ATTRIBUTE_POSITION, 2, GLES20.GL_FLOAT, false)
.add(ShaderProgramConstants.ATTRIBUTE_COLOR_LOCATION, ShaderProgramConstants.ATTRIBUTE_COLOR, 4, GLES20.GL_UNSIGNED_BYTE, true)
.add(ShaderProgramConstants.ATTRIBUTE_TEXTURECOORDINATES_LOCATION, ShaderProgramConstants.ATTRIBUTE_TEXTURECOORDINATES, 2, GLES20.GL_FLOAT, false)
.build();
// ===========================================================
// Fields
// ===========================================================
protected final ITexturedMeshVertexBufferObject mMeshVertexBufferObject;
private int mVertexCountToDraw;
private int mDrawMode;
protected ITextureRegion mTextureRegion;
// ===========================================================
// Constructors
// ===========================================================
/**
* Uses a default {@link HighPerformanceTexturedMeshVertexBufferObject} in {@link DrawType#STATIC} with the {@link VertexBufferObjectAttribute}s: {@link Mesh#VERTEXBUFFEROBJECTATTRIBUTES_DEFAULT}.
*/
public TexturedMesh(final float pX, final float pY, final float[] pBufferData, final int pVertexCount, final DrawMode pDrawMode, final VertexBufferObjectManager pVertexBufferObjectManager) {
this(pX, pY, pBufferData, pVertexCount, pDrawMode, null, pVertexBufferObjectManager, DrawType.STATIC);
}
/**
* Uses a default {@link HighPerformanceTexturedMeshVertexBufferObject} in {@link DrawType#STATIC} with the {@link VertexBufferObjectAttribute}s: {@link Mesh#VERTEXBUFFEROBJECTATTRIBUTES_DEFAULT}.
*/
public TexturedMesh(final float pX, final float pY, final float[] pBufferData, final int pVertexCount, final DrawMode pDrawMode, final ITextureRegion pTextureRegion, final VertexBufferObjectManager pVertexBufferObjectManager) {
this(pX, pY, pBufferData, pVertexCount, pDrawMode, pTextureRegion, pVertexBufferObjectManager, DrawType.STATIC);
}
/**
* Uses a default {@link HighPerformanceTexturedMeshVertexBufferObject} with the {@link VertexBufferObjectAttribute}s: {@link Mesh#VERTEXBUFFEROBJECTATTRIBUTES_DEFAULT}.
*/
public TexturedMesh(final float pX, final float pY, final float[] pBufferData, final int pVertexCount, final DrawMode pDrawMode, final VertexBufferObjectManager pVertexBufferObjectManager, final DrawType pDrawType) {
this(pX, pY, pBufferData, pVertexCount, pDrawMode, null, pVertexBufferObjectManager, pDrawType);
}
/**
* Uses a default {@link HighPerformanceTexturedMeshVertexBufferObject} with the {@link VertexBufferObjectAttribute}s: {@link Mesh#VERTEXBUFFEROBJECTATTRIBUTES_DEFAULT}.
*/
public TexturedMesh(final float pX, final float pY, final float[] pBufferData, final int pVertexCount, final DrawMode pDrawMode, final ITextureRegion pTextureRegion, final VertexBufferObjectManager pVertexBufferObjectManager, final DrawType pDrawType) {
this(pX, pY, pVertexCount, pDrawMode, pTextureRegion, new HighPerformanceTexturedMeshVertexBufferObject(pVertexBufferObjectManager, pBufferData, pVertexCount, pDrawType, true, TexturedMesh.VERTEXBUFFEROBJECTATTRIBUTES_DEFAULT));
}
/**
* Uses a default {@link HighPerformanceTexturedMeshVertexBufferObject} with the {@link VertexBufferObjectAttribute}s: {@link Mesh#VERTEXBUFFEROBJECTATTRIBUTES_DEFAULT}.
*/
public TexturedMesh(final float pX, final float pY, final float[] pVertexX, final float[] pVertexY, final DrawMode pDrawMode, final ITextureRegion pTextureRegion, final VertexBufferObjectManager pVertexBufferObjectManager, final DrawType pDrawType) {
this(pX, pY, pVertexX.length, pDrawMode, pTextureRegion, new HighPerformanceTexturedMeshVertexBufferObject(pVertexBufferObjectManager, buildVertexList(pVertexX, pVertexY), pVertexX.length, pDrawType, true, TexturedMesh.VERTEXBUFFEROBJECTATTRIBUTES_DEFAULT));
}
public TexturedMesh(final float pX, final float pY, final int pVertexCount, final DrawMode pDrawMode, final ITextureRegion pTextureRegion, final ITexturedMeshVertexBufferObject pMeshVertexBufferObject) {
super(pX, pY, PositionColorTextureCoordinatesShaderProgram.getInstance());
this.mDrawMode = pDrawMode.getDrawMode();
this.mTextureRegion = pTextureRegion;
this.mMeshVertexBufferObject = pMeshVertexBufferObject;
this.mVertexCountToDraw = pVertexCount;
if( pTextureRegion != null)
{
this.setBlendingEnabled(true);
this.initBlendFunction(pTextureRegion);
this.onUpdateTextureCoordinates();
}
this.onUpdateVertices();
this.onUpdateColor();
this.mMeshVertexBufferObject.setDirtyOnHardware();
this.setBlendingEnabled(true);
}
// ===========================================================
// Getter & Setter
// ===========================================================
public float[] getBufferData() {
return this.mMeshVertexBufferObject.getBufferData();
}
public void setVertexCountToDraw(final int pVertexCountToDraw) {
this.mVertexCountToDraw = pVertexCountToDraw;
}
public void setDrawMode(final DrawMode pDrawMode) {
this.mDrawMode = pDrawMode.mDrawMode;
}
public ITextureRegion getTextureRegion() {
return this.mTextureRegion;
}
// ===========================================================
// Methods for/from SuperClass/Interfaces
// ===========================================================
@Override
public ITexturedMeshVertexBufferObject getVertexBufferObject() {
return this.mMeshVertexBufferObject;
}
@Override
public void reset() {
super.reset();
this.initBlendFunction(this.getTextureRegion().getTexture());
}
@Override
protected void preDraw(final GLState pGLState, final Camera pCamera) {
super.preDraw(pGLState, pCamera);
this.mTextureRegion.getTexture().bind(pGLState);
this.mMeshVertexBufferObject.bind(pGLState, this.mShaderProgram);
}
@Override
protected void draw(final GLState pGLState, final Camera pCamera) {
this.mMeshVertexBufferObject.draw(this.mDrawMode, this.mVertexCountToDraw);
}
@Override
protected void postDraw(final GLState pGLState, final Camera pCamera) {
this.mMeshVertexBufferObject.unbind(pGLState, this.mShaderProgram);
super.postDraw(pGLState, pCamera);
}
@Override
protected void onUpdateColor() {
this.mMeshVertexBufferObject.onUpdateColor(this);
}
@Override
protected void onUpdateVertices() {
this.mMeshVertexBufferObject.onUpdateVertices(this);
}
protected void onUpdateTextureCoordinates() {
this.mMeshVertexBufferObject.onUpdateTextureCoordinates(this);
}
@Override
@Deprecated
public boolean contains(final float pX, final float pY) {
throw new MethodNotSupportedException();
}
@Override
public boolean collidesWith(final IShape pOtherShape) {
if(pOtherShape instanceof Line) {
// TODO
return false;
} else if(pOtherShape instanceof RectangularShape) {
// TODO
return false;
} else {
return false;
}
}
// ===========================================================
// Methods
// ===========================================================
protected static float[] buildVertexList(float[] pVertexX, float[] pVertexY)
{
assert( pVertexX.length == pVertexY.length );
float[] bufferData = new float[TexturedMesh.VERTEX_SIZE * pVertexX.length];
updateVertexList(pVertexX, pVertexY, bufferData);
return bufferData;
}
protected static void updateVertexList(float[] pVertexX, float[] pVertexY, float[] pBufferData)
{
for( int i = 0; i < pVertexX.length; i++)
{
pBufferData[(i * TexturedMesh.VERTEX_SIZE) + TexturedMesh.VERTEX_INDEX_X] = pVertexX[i];
pBufferData[(i * TexturedMesh.VERTEX_SIZE) + TexturedMesh.VERTEX_INDEX_Y] = pVertexY[i];
}
}
// ===========================================================
// Inner and Anonymous Classes
// ===========================================================
public static interface ITexturedMeshVertexBufferObject extends IVertexBufferObject {
// ===========================================================
// Constants
// ===========================================================
// ===========================================================
// Methods
// ===========================================================
public float[] getBufferData();
public void onUpdateColor(final TexturedMesh pMesh);
public void onUpdateVertices(final TexturedMesh pMesh);
public void onUpdateTextureCoordinates(final TexturedMesh pMesh);
}
public static class HighPerformanceTexturedMeshVertexBufferObject extends HighPerformanceVertexBufferObject implements ITexturedMeshVertexBufferObject {
// ===========================================================
// Constants
// ===========================================================
// ===========================================================
// Fields
// ===========================================================
private final int mVertexCount;
// ===========================================================
// Constructors
// ===========================================================
public HighPerformanceTexturedMeshVertexBufferObject(final VertexBufferObjectManager pVertexBufferObjectManager, final float[] pBufferData, final int pVertexCount, final DrawType pDrawType, final boolean pAutoDispose, final VertexBufferObjectAttributes pVertexBufferObjectAttributes) {
super(pVertexBufferObjectManager, pBufferData, pDrawType, pAutoDispose, pVertexBufferObjectAttributes);
this.mVertexCount = pVertexCount;
}
// ===========================================================
// Getter & Setter
// ===========================================================
// ===========================================================
// Methods for/from SuperClass/Interfaces
// ===========================================================
@Override
public void onUpdateColor(final TexturedMesh pMesh) {
final float[] bufferData = this.mBufferData;
final float packedColor = pMesh.getColor().getABGRPackedFloat();
for(int i = 0; i < this.mVertexCount; i++) {
bufferData[(i * TexturedMesh.VERTEX_SIZE) + TexturedMesh.COLOR_INDEX] = packedColor;
}
this.setDirtyOnHardware();
}
@Override
public void onUpdateVertices(final TexturedMesh pMesh) {
/* Since the buffer data is managed from the caller, we just mark the buffer data as dirty. */
this.setDirtyOnHardware();
}
@Override
public void onUpdateTextureCoordinates(final TexturedMesh pMesh) {
final float[] bufferData = this.mBufferData;
final ITextureRegion textureRegion = pMesh.getTextureRegion(); // TODO Optimize with field access?
float textureWidth = textureRegion.getWidth();
float textureHeight = textureRegion.getHeight();
// x0 is mapped to u0
// y0 is mapped to v0
// TODO get initial mapping
float x0 = 0; // pMesh.getX0();
float y0 = 0; //pMesh.getY0();
for(int i = 0; i < this.mVertexCount; i++) {
float x = bufferData[(i * TexturedMesh.VERTEX_SIZE) + TexturedMesh.VERTEX_INDEX_X];
float y = bufferData[(i * TexturedMesh.VERTEX_SIZE) + TexturedMesh.VERTEX_INDEX_Y];
float u = (x - x0) / textureWidth;
float v = (y - y0) / textureHeight;
Debug.d("u = " + u );
Debug.d("v = " + v );
Debug.d("x = " + x );
Debug.d("y = " + y );
bufferData[(i * TexturedMesh.VERTEX_SIZE) + TexturedMesh.TEXTURECOORDINATES_INDEX_U] = u;
bufferData[(i * TexturedMesh.VERTEX_SIZE) + TexturedMesh.TEXTURECOORDINATES_INDEX_V] = v;
}
Debug.d("v ---" );
this.setDirtyOnHardware();
}
// ===========================================================
// Methods
// ===========================================================
// ===========================================================
// Inner and Anonymous Classes
// ===========================================================
}
public static enum DrawMode {
// ===========================================================
// Elements
// ===========================================================
POINTS(GLES20.GL_POINTS),
LINE_STRIP(GLES20.GL_LINE_STRIP),
LINE_LOOP(GLES20.GL_LINE_LOOP),
LINES(GLES20.GL_LINES),
TRIANGLE_STRIP(GLES20.GL_TRIANGLE_STRIP),
TRIANGLE_FAN(GLES20.GL_TRIANGLE_FAN),
TRIANGLES(GLES20.GL_TRIANGLES);
// ===========================================================
// Constants
// ===========================================================
public final int mDrawMode;
// ===========================================================
// Fields
// ===========================================================
// ===========================================================
// Constructors
// ===========================================================
private DrawMode(final int pDrawMode) {
this.mDrawMode = pDrawMode;
}
// ===========================================================
// Getter & Setter
// ===========================================================
public int getDrawMode() {
return this.mDrawMode;
}
// ===========================================================
// Methods for/from SuperClass/Interfaces
// ===========================================================
// ===========================================================
// Methods
// ===========================================================
// ===========================================================
// Inner and Anonymous Classes
// ===========================================================
}
}