parent
c03254ae24
commit
c2a93b6ba1
@ -0,0 +1,414 @@
|
||||
/*
|
||||
* Copyright (C) 2012 CyberAgent
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
package com.seu.magicfilter.base.gpuimage;
|
||||
|
||||
import android.content.Context;
|
||||
import android.graphics.PointF;
|
||||
import android.opengl.GLES11Ext;
|
||||
import android.opengl.GLES20;
|
||||
|
||||
import com.seu.magicfilter.utils.MagicFilterType;
|
||||
import com.seu.magicfilter.utils.OpenGLUtils;
|
||||
|
||||
import net.ossrs.yasea.R;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.ByteOrder;
|
||||
import java.nio.FloatBuffer;
|
||||
import java.nio.IntBuffer;
|
||||
import java.util.LinkedList;
|
||||
|
||||
public class GPUImageFilter {
|
||||
|
||||
private boolean mIsInitialized;
|
||||
private Context mContext;
|
||||
private MagicFilterType mType = MagicFilterType.NONE;
|
||||
private final LinkedList<Runnable> mRunOnDraw;
|
||||
private final int mVertexShaderId;
|
||||
private final int mFragmentShaderId;
|
||||
|
||||
private int mGLProgId;
|
||||
private int mGLPositionIndex;
|
||||
private int mGLInputImageTextureIndex;
|
||||
private int mGLTextureCoordinateIndex;
|
||||
private int mGLTextureTransformIndex;
|
||||
|
||||
protected int mInputWidth;
|
||||
protected int mInputHeight;
|
||||
protected int mOutputWidth;
|
||||
protected int mOutputHeight;
|
||||
protected FloatBuffer mGLCubeBuffer;
|
||||
protected FloatBuffer mGLTextureBuffer;
|
||||
|
||||
private int[] mGLCubeId;
|
||||
private int[] mGLTextureCoordinateId;
|
||||
private float[] mGLTextureTransformMatrix;
|
||||
|
||||
private int[] mGLFboId;
|
||||
private int[] mGLFboTexId;
|
||||
private IntBuffer mGLFboBuffer;
|
||||
|
||||
public GPUImageFilter() {
|
||||
this(MagicFilterType.NONE);
|
||||
}
|
||||
|
||||
public GPUImageFilter(MagicFilterType type) {
|
||||
this(type, R.raw.vertex, R.raw.fragment);
|
||||
}
|
||||
|
||||
public GPUImageFilter(MagicFilterType type, int fragmentShaderId) {
|
||||
this(type, R.raw.vertex, fragmentShaderId);
|
||||
}
|
||||
|
||||
public GPUImageFilter(MagicFilterType type, int vertexShaderId, int fragmentShaderId) {
|
||||
mType = type;
|
||||
mRunOnDraw = new LinkedList<>();
|
||||
mVertexShaderId = vertexShaderId;
|
||||
mFragmentShaderId = fragmentShaderId;
|
||||
}
|
||||
|
||||
public void init(Context context) {
|
||||
mContext = context;
|
||||
onInit();
|
||||
onInitialized();
|
||||
}
|
||||
|
||||
protected void onInit() {
|
||||
initVbo();
|
||||
loadSamplerShader();
|
||||
}
|
||||
|
||||
protected void onInitialized() {
|
||||
mIsInitialized = true;
|
||||
}
|
||||
|
||||
public final void destroy() {
|
||||
mIsInitialized = false;
|
||||
destroyFboTexture();
|
||||
destoryVbo();
|
||||
GLES20.glDeleteProgram(mGLProgId);
|
||||
onDestroy();
|
||||
}
|
||||
|
||||
protected void onDestroy() {
|
||||
}
|
||||
|
||||
public void onInputSizeChanged(final int width, final int height) {
|
||||
mInputWidth = width;
|
||||
mInputHeight = height;
|
||||
initFboTexture(width, height);
|
||||
}
|
||||
|
||||
public void onDisplaySizeChanged(final int width, final int height) {
|
||||
mOutputWidth = width;
|
||||
mOutputHeight = height;
|
||||
}
|
||||
|
||||
private void loadSamplerShader() {
|
||||
mGLProgId = OpenGLUtils.loadProgram(OpenGLUtils.readShaderFromRawResource(getContext(), mVertexShaderId),
|
||||
OpenGLUtils.readShaderFromRawResource(getContext(), mFragmentShaderId));
|
||||
mGLPositionIndex = GLES20.glGetAttribLocation(mGLProgId, "position");
|
||||
mGLTextureCoordinateIndex = GLES20.glGetAttribLocation(mGLProgId,"inputTextureCoordinate");
|
||||
mGLTextureTransformIndex = GLES20.glGetUniformLocation(mGLProgId, "textureTransform");
|
||||
mGLInputImageTextureIndex = GLES20.glGetUniformLocation(mGLProgId, "inputImageTexture");
|
||||
}
|
||||
|
||||
private void initVbo() {
|
||||
final float VEX_CUBE[] = {
|
||||
-1.0f, -1.0f, // Bottom left.
|
||||
1.0f, -1.0f, // Bottom right.
|
||||
-1.0f, 1.0f, // Top left.
|
||||
1.0f, 1.0f, // Top right.
|
||||
};
|
||||
|
||||
final float TEX_COORD[] = {
|
||||
0.0f, 0.0f, // Bottom left.
|
||||
1.0f, 0.0f, // Bottom right.
|
||||
0.0f, 1.0f, // Top left.
|
||||
1.0f, 1.0f // Top right.
|
||||
};
|
||||
|
||||
mGLCubeBuffer = ByteBuffer.allocateDirect(VEX_CUBE.length * 4)
|
||||
.order(ByteOrder.nativeOrder()).asFloatBuffer();
|
||||
mGLCubeBuffer.put(VEX_CUBE).position(0);
|
||||
|
||||
mGLTextureBuffer = ByteBuffer.allocateDirect(TEX_COORD.length * 4)
|
||||
.order(ByteOrder.nativeOrder()).asFloatBuffer();
|
||||
mGLTextureBuffer.put(TEX_COORD).position(0);
|
||||
|
||||
mGLCubeId = new int[1];
|
||||
mGLTextureCoordinateId = new int[1];
|
||||
|
||||
GLES20.glGenBuffers(1, mGLCubeId, 0);
|
||||
GLES20.glBindBuffer(GLES20.GL_ARRAY_BUFFER, mGLCubeId[0]);
|
||||
GLES20.glBufferData(GLES20.GL_ARRAY_BUFFER, mGLCubeBuffer.capacity() * 4, mGLCubeBuffer, GLES20.GL_STATIC_DRAW);
|
||||
|
||||
GLES20.glGenBuffers(1, mGLTextureCoordinateId, 0);
|
||||
GLES20.glBindBuffer(GLES20.GL_ARRAY_BUFFER, mGLTextureCoordinateId[0]);
|
||||
GLES20.glBufferData(GLES20.GL_ARRAY_BUFFER, mGLTextureBuffer.capacity() * 4, mGLTextureBuffer, GLES20.GL_STATIC_DRAW);
|
||||
}
|
||||
|
||||
private void destoryVbo() {
|
||||
if (mGLCubeId != null) {
|
||||
GLES20.glDeleteBuffers(1, mGLCubeId, 0);
|
||||
mGLCubeId = null;
|
||||
}
|
||||
if (mGLTextureCoordinateId != null) {
|
||||
GLES20.glDeleteBuffers(1, mGLTextureCoordinateId, 0);
|
||||
mGLTextureCoordinateId = null;
|
||||
}
|
||||
}
|
||||
|
||||
private void initFboTexture(int width, int height) {
|
||||
if (mGLFboId != null && (mInputWidth != width || mInputHeight != height)) {
|
||||
destroyFboTexture();
|
||||
}
|
||||
|
||||
mGLFboId = new int[1];
|
||||
mGLFboTexId = new int[1];
|
||||
mGLFboBuffer = IntBuffer.allocate(width * height);
|
||||
|
||||
GLES20.glGenFramebuffers(1, mGLFboId, 0);
|
||||
GLES20.glGenTextures(1, mGLFboTexId, 0);
|
||||
GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, mGLFboTexId[0]);
|
||||
GLES20.glTexImage2D(GLES20.GL_TEXTURE_2D, 0, GLES20.GL_RGBA, width, height, 0, GLES20.GL_RGBA, GLES20.GL_UNSIGNED_BYTE, null);
|
||||
GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MAG_FILTER, GLES20.GL_LINEAR);
|
||||
GLES20.glTexParameterf(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MIN_FILTER, GLES20.GL_LINEAR);
|
||||
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);
|
||||
GLES20.glBindFramebuffer(GLES20.GL_FRAMEBUFFER, mGLFboId[0]);
|
||||
GLES20.glFramebufferTexture2D(GLES20.GL_FRAMEBUFFER, GLES20.GL_COLOR_ATTACHMENT0, GLES20.GL_TEXTURE_2D, mGLFboTexId[0], 0);
|
||||
GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, 0);
|
||||
GLES20.glBindFramebuffer(GLES20.GL_FRAMEBUFFER, 0);
|
||||
}
|
||||
|
||||
private void destroyFboTexture() {
|
||||
if (mGLFboTexId != null) {
|
||||
GLES20.glDeleteTextures(1, mGLFboTexId, 0);
|
||||
mGLFboTexId = null;
|
||||
}
|
||||
if (mGLFboId != null) {
|
||||
GLES20.glDeleteFramebuffers(1, mGLFboId, 0);
|
||||
mGLFboId = null;
|
||||
}
|
||||
}
|
||||
|
||||
public int onDrawFrame(final int textureId, final FloatBuffer cubeBuffer, final FloatBuffer textureBuffer) {
|
||||
if (!mIsInitialized) {
|
||||
return OpenGLUtils.NOT_INIT;
|
||||
}
|
||||
|
||||
GLES20.glUseProgram(mGLProgId);
|
||||
runPendingOnDrawTasks();
|
||||
|
||||
GLES20.glEnableVertexAttribArray(mGLPositionIndex);
|
||||
GLES20.glVertexAttribPointer(mGLPositionIndex, 2, GLES20.GL_FLOAT, false, 4 * 2, cubeBuffer);
|
||||
|
||||
GLES20.glEnableVertexAttribArray(mGLTextureCoordinateIndex);
|
||||
GLES20.glVertexAttribPointer(mGLTextureCoordinateIndex, 2, GLES20.GL_FLOAT, false, 4 * 2, textureBuffer);
|
||||
|
||||
if (textureId != OpenGLUtils.NO_TEXTURE) {
|
||||
GLES20.glActiveTexture(GLES20.GL_TEXTURE0);
|
||||
GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, textureId);
|
||||
GLES20.glUniform1i(mGLInputImageTextureIndex, 0);
|
||||
}
|
||||
|
||||
onDrawArraysPre();
|
||||
GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, 4);
|
||||
onDrawArraysAfter();
|
||||
|
||||
GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, 0);
|
||||
|
||||
GLES20.glDisableVertexAttribArray(mGLPositionIndex);
|
||||
GLES20.glDisableVertexAttribArray(mGLTextureCoordinateIndex);
|
||||
|
||||
return OpenGLUtils.ON_DRAWN;
|
||||
}
|
||||
|
||||
public int onDrawFrame(int cameraTextureId) {
|
||||
if (!mIsInitialized) {
|
||||
return OpenGLUtils.NOT_INIT;
|
||||
}
|
||||
|
||||
if (mGLFboId == null) {
|
||||
return OpenGLUtils.NO_TEXTURE;
|
||||
}
|
||||
|
||||
GLES20.glUseProgram(mGLProgId);
|
||||
runPendingOnDrawTasks();
|
||||
|
||||
GLES20.glBindBuffer(GLES20.GL_ARRAY_BUFFER, mGLCubeId[0]);
|
||||
GLES20.glEnableVertexAttribArray(mGLPositionIndex);
|
||||
GLES20.glVertexAttribPointer(mGLPositionIndex, 2, GLES20.GL_FLOAT, false, 4 * 2, 0);
|
||||
|
||||
GLES20.glBindBuffer(GLES20.GL_ARRAY_BUFFER, mGLTextureCoordinateId[0]);
|
||||
GLES20.glEnableVertexAttribArray(mGLTextureCoordinateIndex);
|
||||
GLES20.glVertexAttribPointer(mGLTextureCoordinateIndex, 2, GLES20.GL_FLOAT, false, 4 * 2, 0);
|
||||
|
||||
GLES20.glUniformMatrix4fv(mGLTextureTransformIndex, 1, false, mGLTextureTransformMatrix, 0);
|
||||
|
||||
GLES20.glActiveTexture(GLES20.GL_TEXTURE0);
|
||||
GLES20.glBindTexture(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, cameraTextureId);
|
||||
GLES20.glUniform1i(mGLInputImageTextureIndex, 0);
|
||||
|
||||
onDrawArraysPre();
|
||||
|
||||
GLES20.glViewport(0, 0, mInputWidth, mInputHeight);
|
||||
GLES20.glBindFramebuffer(GLES20.GL_FRAMEBUFFER, mGLFboId[0]);
|
||||
GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, 4);
|
||||
GLES20.glReadPixels(0, 0, mInputWidth, mInputHeight, GLES20.GL_RGBA, GLES20.GL_UNSIGNED_BYTE, mGLFboBuffer);
|
||||
GLES20.glBindFramebuffer(GLES20.GL_FRAMEBUFFER, 0);
|
||||
GLES20.glViewport(0, 0, mOutputWidth, mOutputHeight);
|
||||
|
||||
GLES20.glDrawArrays(GLES20.GL_TRIANGLE_STRIP, 0, 4);
|
||||
|
||||
onDrawArraysAfter();
|
||||
|
||||
GLES20.glBindTexture(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, 0);
|
||||
|
||||
GLES20.glDisableVertexAttribArray(mGLPositionIndex);
|
||||
GLES20.glDisableVertexAttribArray(mGLTextureCoordinateIndex);
|
||||
|
||||
GLES20.glBindBuffer(GLES20.GL_ARRAY_BUFFER, 0);
|
||||
|
||||
return mGLFboTexId[0];
|
||||
}
|
||||
|
||||
protected void onDrawArraysPre() {}
|
||||
|
||||
protected void onDrawArraysAfter() {}
|
||||
|
||||
private void runPendingOnDrawTasks() {
|
||||
while (!mRunOnDraw.isEmpty()) {
|
||||
mRunOnDraw.removeFirst().run();
|
||||
}
|
||||
}
|
||||
|
||||
public int getProgram() {
|
||||
return mGLProgId;
|
||||
}
|
||||
|
||||
public IntBuffer getGLFboBuffer() {
|
||||
return mGLFboBuffer;
|
||||
}
|
||||
|
||||
protected Context getContext() {
|
||||
return mContext;
|
||||
}
|
||||
|
||||
protected MagicFilterType getFilterType() {
|
||||
return mType;
|
||||
}
|
||||
|
||||
public void setTextureTransformMatrix(float[] mtx){
|
||||
mGLTextureTransformMatrix = mtx;
|
||||
}
|
||||
|
||||
protected void setInteger(final int location, final int intValue) {
|
||||
runOnDraw(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
GLES20.glUniform1i(location, intValue);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void setFloat(final int location, final float floatValue) {
|
||||
runOnDraw(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
GLES20.glUniform1f(location, floatValue);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void setFloatVec2(final int location, final float[] arrayValue) {
|
||||
runOnDraw(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
GLES20.glUniform2fv(location, 1, FloatBuffer.wrap(arrayValue));
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void setFloatVec3(final int location, final float[] arrayValue) {
|
||||
runOnDraw(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
GLES20.glUniform3fv(location, 1, FloatBuffer.wrap(arrayValue));
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void setFloatVec4(final int location, final float[] arrayValue) {
|
||||
runOnDraw(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
GLES20.glUniform4fv(location, 1, FloatBuffer.wrap(arrayValue));
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void setFloatArray(final int location, final float[] arrayValue) {
|
||||
runOnDraw(new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
GLES20.glUniform1fv(location, arrayValue.length, FloatBuffer.wrap(arrayValue));
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void setPoint(final int location, final PointF point) {
|
||||
runOnDraw(new Runnable() {
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
float[] vec2 = new float[2];
|
||||
vec2[0] = point.x;
|
||||
vec2[1] = point.y;
|
||||
GLES20.glUniform2fv(location, 1, vec2, 0);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void setUniformMatrix3f(final int location, final float[] matrix) {
|
||||
runOnDraw(new Runnable() {
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
GLES20.glUniformMatrix3fv(location, 1, false, matrix, 0);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void setUniformMatrix4f(final int location, final float[] matrix) {
|
||||
runOnDraw(new Runnable() {
|
||||
|
||||
@Override
|
||||
public void run() {
|
||||
GLES20.glUniformMatrix4fv(location, 1, false, matrix, 0);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void runOnDraw(final Runnable runnable) {
|
||||
synchronized (mRunOnDraw) {
|
||||
mRunOnDraw.addLast(runnable);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1,54 +0,0 @@
|
||||
/*
|
||||
* Copyright (C) 2012 CyberAgent
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
package com.seu.magicfilter.utils;
|
||||
|
||||
public enum Rotation {
|
||||
NORMAL, ROTATION_90, ROTATION_180, ROTATION_270;
|
||||
|
||||
/**
|
||||
* Retrieves the int representation of the Rotation.
|
||||
*
|
||||
* @return 0, 90, 180 or 270
|
||||
*/
|
||||
public int asInt() {
|
||||
switch (this) {
|
||||
case NORMAL: return 0;
|
||||
case ROTATION_90: return 90;
|
||||
case ROTATION_180: return 180;
|
||||
case ROTATION_270: return 270;
|
||||
default: throw new IllegalStateException("Unknown Rotation!");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a Rotation from an integer. Needs to be either 0, 90, 180 or 270.
|
||||
*
|
||||
* @param rotation 0, 90, 180 or 270
|
||||
* @return Rotation object
|
||||
*/
|
||||
public static Rotation fromInt(int rotation) {
|
||||
switch (rotation) {
|
||||
case 0: return NORMAL;
|
||||
case 90: return ROTATION_90;
|
||||
case 180: return ROTATION_180;
|
||||
case 270: return ROTATION_270;
|
||||
case 360: return NORMAL;
|
||||
default: throw new IllegalStateException(
|
||||
rotation + " is an unknown rotation. Needs to be either 0, 90, 180 or 270!");
|
||||
}
|
||||
}
|
||||
}
|
@ -1,96 +0,0 @@
|
||||
/*
|
||||
* Copyright (C) 2012 CyberAgent
|
||||
*
|
||||
* 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.
|
||||
*/
|
||||
|
||||
package com.seu.magicfilter.utils;
|
||||
|
||||
public class TextureRotationUtil {
|
||||
|
||||
public static final float TEXTURE_NO_ROTATION[] = {
|
||||
0.0f, 1.0f,
|
||||
1.0f, 1.0f,
|
||||
0.0f, 0.0f,
|
||||
1.0f, 0.0f,
|
||||
};
|
||||
|
||||
public static final float TEXTURE_ROTATED_90[] = {
|
||||
1.0f, 1.0f,
|
||||
1.0f, 0.0f,
|
||||
0.0f, 1.0f,
|
||||
0.0f, 0.0f,
|
||||
};
|
||||
public static final float TEXTURE_ROTATED_180[] = {
|
||||
1.0f, 0.0f,
|
||||
0.0f, 0.0f,
|
||||
1.0f, 1.0f,
|
||||
0.0f, 1.0f,
|
||||
};
|
||||
public static final float TEXTURE_ROTATED_270[] = {
|
||||
0.0f, 0.0f,
|
||||
0.0f, 1.0f,
|
||||
1.0f, 0.0f,
|
||||
1.0f, 1.0f,
|
||||
};
|
||||
|
||||
public static final float CUBE[] = {
|
||||
-1.0f, -1.0f,
|
||||
1.0f, -1.0f,
|
||||
-1.0f, 1.0f,
|
||||
1.0f, 1.0f,
|
||||
};
|
||||
|
||||
private TextureRotationUtil() {}
|
||||
|
||||
public static float[] getRotation(final Rotation rotation, final boolean flipHorizontal,
|
||||
final boolean flipVertical) {
|
||||
float[] rotatedTex;
|
||||
switch (rotation) {
|
||||
case ROTATION_90:
|
||||
rotatedTex = TEXTURE_ROTATED_90;
|
||||
break;
|
||||
case ROTATION_180:
|
||||
rotatedTex = TEXTURE_ROTATED_180;
|
||||
break;
|
||||
case ROTATION_270:
|
||||
rotatedTex = TEXTURE_ROTATED_270;
|
||||
break;
|
||||
case NORMAL:
|
||||
default:
|
||||
rotatedTex = TEXTURE_NO_ROTATION;
|
||||
break;
|
||||
}
|
||||
if (flipHorizontal) {
|
||||
rotatedTex = new float[]{
|
||||
flip(rotatedTex[0]), rotatedTex[1],
|
||||
flip(rotatedTex[2]), rotatedTex[3],
|
||||
flip(rotatedTex[4]), rotatedTex[5],
|
||||
flip(rotatedTex[6]), rotatedTex[7],
|
||||
};
|
||||
}
|
||||
if (flipVertical) {
|
||||
rotatedTex = new float[]{
|
||||
rotatedTex[0], flip(rotatedTex[1]),
|
||||
rotatedTex[2], flip(rotatedTex[3]),
|
||||
rotatedTex[4], flip(rotatedTex[5]),
|
||||
rotatedTex[6], flip(rotatedTex[7]),
|
||||
};
|
||||
}
|
||||
return rotatedTex;
|
||||
}
|
||||
|
||||
private static float flip(final float i) {
|
||||
return i == 0.0f ? 1.0f : 0.0f;
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue