/**
* Copyright 2008 - 2015 The Loon Game Engine Authors
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*
* @project loon
* @author cping
* @email:javachenpeng@yahoo.com
* @version 0.5
*/
package loon.opengl;
import java.nio.ByteBuffer;
import java.nio.FloatBuffer;
import loon.LSystem;
import loon.Support;
public class VertexArray implements VertexData {
final VertexAttributes attributes;
final FloatBuffer buffer;
final ByteBuffer byteBuffer;
boolean isBound = false;
public VertexArray(int numVertices, VertexAttribute... attributes) {
this(numVertices, new VertexAttributes(attributes));
}
public VertexArray(int numVertices, VertexAttributes attributes) {
this.attributes = attributes;
byteBuffer = LSystem.base().support()
.newUnsafeByteBuffer(this.attributes.vertexSize * numVertices);
buffer = byteBuffer.asFloatBuffer();
buffer.flip();
byteBuffer.flip();
}
@Override
public FloatBuffer getBuffer() {
return buffer;
}
@Override
public int getNumVertices() {
return buffer.limit() * 4 / attributes.vertexSize;
}
@Override
public int getNumMaxVertices() {
return byteBuffer.capacity() / attributes.vertexSize;
}
@Override
public void setVertices(float[] vertices, int offset, int count) {
Support support = LSystem.base().support();
if (support.isNative()) {
support.copy(vertices, byteBuffer, offset, count);
} else {
support.copy(vertices, buffer, offset, count);
}
buffer.position(0);
buffer.limit(count);
}
@Override
public void updateVertices(int targetOffset, float[] vertices,
int sourceOffset, int count) {
final int pos = byteBuffer.position();
byteBuffer.position(targetOffset * 4);
LSystem.base().support()
.copy(vertices, byteBuffer, sourceOffset, count);
byteBuffer.position(pos);
}
@Override
public void bind(final ShaderProgram shader) {
bind(shader, null);
}
@Override
public void bind(final ShaderProgram shader, final int[] locations) {
final int numAttributes = attributes.size();
byteBuffer.limit(buffer.limit() * 4);
if (locations == null) {
for (int i = 0; i < numAttributes; i++) {
final VertexAttribute attribute = attributes.get(i);
final int location = shader
.getAttributeLocation(attribute.alias);
if (location < 0) {
continue;
}
shader.enableVertexAttribute(location);
if (attribute.type == GL20.GL_FLOAT) {
buffer.position(attribute.offset / 4);
shader.setVertexAttribute(location,
attribute.numComponents, attribute.type,
attribute.normalized, attributes.vertexSize, buffer);
} else {
byteBuffer.position(attribute.offset);
shader.setVertexAttribute(location,
attribute.numComponents, attribute.type,
attribute.normalized, attributes.vertexSize,
byteBuffer);
}
}
} else {
for (int i = 0; i < numAttributes; i++) {
final VertexAttribute attribute = attributes.get(i);
final int location = locations[i];
if (location < 0) {
continue;
}
shader.enableVertexAttribute(location);
if (attribute.type == GL20.GL_FLOAT) {
buffer.position(attribute.offset / 4);
shader.setVertexAttribute(location,
attribute.numComponents, attribute.type,
attribute.normalized, attributes.vertexSize, buffer);
} else {
byteBuffer.position(attribute.offset);
shader.setVertexAttribute(location,
attribute.numComponents, attribute.type,
attribute.normalized, attributes.vertexSize,
byteBuffer);
}
}
}
isBound = true;
}
@Override
public void unbind(ShaderProgram shader) {
unbind(shader, null);
}
@Override
public void unbind(ShaderProgram shader, int[] locations) {
final int numAttributes = attributes.size();
if (locations == null) {
for (int i = 0; i < numAttributes; i++) {
shader.disableVertexAttribute(attributes.get(i).alias);
}
} else {
for (int i = 0; i < numAttributes; i++) {
final int location = locations[i];
if (location >= 0) {
shader.disableVertexAttribute(location);
}
}
}
isBound = false;
}
@Override
public VertexAttributes getAttributes() {
return attributes;
}
@Override
public void invalidate() {
}
@Override
public void close() {
LSystem.base().support().disposeUnsafeByteBuffer(byteBuffer);
}
}