|
@@ -4,69 +4,59 @@ import java.io.IOException;
|
|
import java.net.URL;
|
|
import java.net.URL;
|
|
import java.nio.FloatBuffer;
|
|
import java.nio.FloatBuffer;
|
|
import java.util.ArrayList;
|
|
import java.util.ArrayList;
|
|
-import java.util.LinkedList;
|
|
|
|
-import java.util.List;
|
|
|
|
import java.util.Scanner;
|
|
import java.util.Scanner;
|
|
import org.lwjgl.BufferUtils;
|
|
import org.lwjgl.BufferUtils;
|
|
import static org.lwjgl.opengl.GL11.*;
|
|
import static org.lwjgl.opengl.GL11.*;
|
|
import static org.lwjgl.opengl.GL14.*;
|
|
import static org.lwjgl.opengl.GL14.*;
|
|
import static org.lwjgl.opengl.GL20.*;
|
|
import static org.lwjgl.opengl.GL20.*;
|
|
|
|
|
|
-public final class Shader
|
|
|
|
|
|
+public final class Renderer
|
|
{
|
|
{
|
|
- private static int program = -1;
|
|
|
|
|
|
+ private final int program;
|
|
|
|
|
|
- public final static int BASE_WIDTH = 1024;
|
|
|
|
- public final static int BASE_HEIGHT = 620;
|
|
|
|
|
|
+ private float width;
|
|
|
|
+ private float height;
|
|
|
|
+ private int scale;
|
|
|
|
|
|
- private static int width = 512;
|
|
|
|
- private static int height = 310;
|
|
|
|
- private static int scale = 2;
|
|
|
|
-
|
|
|
|
- private static final List<Runnable> TASKS = new LinkedList<>();
|
|
|
|
-
|
|
|
|
- protected static boolean initDone = false;
|
|
|
|
-
|
|
|
|
- private final static FontRenderer FONT_RENDERER = new FontRenderer();
|
|
|
|
- private final static ColorRenderer COLOR_RENDERER = new ColorRenderer();
|
|
|
|
- private final static DirectTextureRenderer TEXTURE_RENDERER = new DirectTextureRenderer();
|
|
|
|
|
|
+ private final FontRenderer fontRenderer;
|
|
|
|
+ private final ColorRenderer colorRenderer = new ColorRenderer();
|
|
|
|
+ private final DirectTextureRenderer textureRenderer = new DirectTextureRenderer();
|
|
|
|
|
|
// uniform stuff
|
|
// uniform stuff
|
|
- private static int unifViewMatrix = -1;
|
|
|
|
- private static int unifModelMatrix = -1;
|
|
|
|
- private static MatrixStack modelMatrix;
|
|
|
|
-
|
|
|
|
- private static int unifAmbientLight = -1;
|
|
|
|
- private static int[][] unifLight;
|
|
|
|
-
|
|
|
|
- private static int unifUseTexture = -1;
|
|
|
|
- private static int unifUseColor = -1;
|
|
|
|
- private static int unifUseLight = -1;
|
|
|
|
- private static int unifUseMixColor = -1;
|
|
|
|
- private static int unifMixColorLoc = -1;
|
|
|
|
- private static float[] unifMixColor = new float[] {0.0f, 0.0f, 0.0f, 0.0f};
|
|
|
|
-
|
|
|
|
- protected static void init()
|
|
|
|
|
|
+ private final int unifViewMatrix;
|
|
|
|
+ private final int unifModelMatrix;
|
|
|
|
+ private final MatrixStack modelMatrix = new MatrixStack(20);
|
|
|
|
+
|
|
|
|
+ private final int unifAmbientLight;
|
|
|
|
+ private final int[][] unifLight = new int[32][3];
|
|
|
|
+
|
|
|
|
+ private final int unifUseTexture;
|
|
|
|
+ private final int unifUseColor;
|
|
|
|
+ private final int unifUseLight;
|
|
|
|
+ private final int unifUseMixColor;
|
|
|
|
+ private final int unifMixColorLoc;
|
|
|
|
+ private final float[] unifMixColor = new float[] {0.0f, 0.0f, 0.0f, 0.0f};
|
|
|
|
+
|
|
|
|
+ protected Renderer(int width, int height)
|
|
{
|
|
{
|
|
program = createShaderProgram("vertex.vs", "fragment.fs");
|
|
program = createShaderProgram("vertex.vs", "fragment.fs");
|
|
glUseProgram(program);
|
|
glUseProgram(program);
|
|
|
|
|
|
unifViewMatrix = glGetUniformLocation(program, "viewMatrix");
|
|
unifViewMatrix = glGetUniformLocation(program, "viewMatrix");
|
|
- updateViewMatrix();
|
|
|
|
|
|
+ fontRenderer = new FontRenderer();
|
|
|
|
+ setViewPort(width, height);
|
|
|
|
|
|
unifModelMatrix = glGetUniformLocation(program, "modelMatrix");
|
|
unifModelMatrix = glGetUniformLocation(program, "modelMatrix");
|
|
- modelMatrix = new MatrixStack(20);
|
|
|
|
updateMatrix();
|
|
updateMatrix();
|
|
|
|
|
|
unifAmbientLight = glGetUniformLocation(program, "ambientLight");
|
|
unifAmbientLight = glGetUniformLocation(program, "ambientLight");
|
|
setAmbientLight(1.0f, 1.0f, 1.0f);
|
|
setAmbientLight(1.0f, 1.0f, 1.0f);
|
|
|
|
|
|
- unifLight = new int[32][3];
|
|
|
|
for(int index = 0; index < unifLight.length; index++)
|
|
for(int index = 0; index < unifLight.length; index++)
|
|
{
|
|
{
|
|
- unifLight[index][0] = glGetUniformLocation(program, "lights[" + index + "].color");
|
|
|
|
- unifLight[index][1] = glGetUniformLocation(program, "lights[" + index + "].pos");
|
|
|
|
- unifLight[index][2] = glGetUniformLocation(program, "lights[" + index + "].strength");
|
|
|
|
|
|
+ unifLight[index][0] = glGetUniformLocation(program, String.format("lights[%d].color", index));
|
|
|
|
+ unifLight[index][1] = glGetUniformLocation(program, String.format("lights[%d].pos", index));
|
|
|
|
+ unifLight[index][2] = glGetUniformLocation(program, String.format("lights[%d].strength", index));
|
|
setLightColor(index, 0.0f, 0.0f, 0.0f);
|
|
setLightColor(index, 0.0f, 0.0f, 0.0f);
|
|
setLightLocation(index, 0.0f, 0.0f);
|
|
setLightLocation(index, 0.0f, 0.0f);
|
|
setLightStrength(index, 0.0f);
|
|
setLightStrength(index, 0.0f);
|
|
@@ -86,45 +76,25 @@ public final class Shader
|
|
unifMixColorLoc = glGetUniformLocation(program, "mixColor");
|
|
unifMixColorLoc = glGetUniformLocation(program, "mixColor");
|
|
setMixColor(0.0f, 0.0f, 0.0f, 0.0f);
|
|
setMixColor(0.0f, 0.0f, 0.0f, 0.0f);
|
|
|
|
|
|
- setViewPort(BASE_WIDTH, BASE_HEIGHT);
|
|
|
|
- initDone = true;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- protected static int getProgram()
|
|
|
|
- {
|
|
|
|
- return program;
|
|
|
|
|
|
+ setViewPort(width, height);
|
|
}
|
|
}
|
|
|
|
|
|
- protected static void addTask(Runnable r)
|
|
|
|
|
|
+ public FontRenderer getFontRenderer()
|
|
{
|
|
{
|
|
- TASKS.add(r);
|
|
|
|
|
|
+ return fontRenderer;
|
|
}
|
|
}
|
|
|
|
|
|
- protected static void doTasks()
|
|
|
|
|
|
+ public ColorRenderer getColorRenderer()
|
|
{
|
|
{
|
|
- if(!TASKS.isEmpty())
|
|
|
|
- {
|
|
|
|
- TASKS.forEach(r -> r.run());
|
|
|
|
- TASKS.clear();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- public static FontRenderer getFontRenderer()
|
|
|
|
- {
|
|
|
|
- return FONT_RENDERER;
|
|
|
|
|
|
+ return colorRenderer;
|
|
}
|
|
}
|
|
|
|
|
|
- public static ColorRenderer getColorRenderer()
|
|
|
|
|
|
+ public DirectTextureRenderer getTextureRenderer()
|
|
{
|
|
{
|
|
- return COLOR_RENDERER;
|
|
|
|
|
|
+ return textureRenderer;
|
|
}
|
|
}
|
|
|
|
|
|
- public static DirectTextureRenderer getTextureRenderer()
|
|
|
|
- {
|
|
|
|
- return TEXTURE_RENDERER;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- private static void updateViewMatrix()
|
|
|
|
|
|
+ private void updateViewMatrix()
|
|
{
|
|
{
|
|
FloatBuffer buffer = BufferUtils.createFloatBuffer(16);
|
|
FloatBuffer buffer = BufferUtils.createFloatBuffer(16);
|
|
|
|
|
|
@@ -153,117 +123,119 @@ public final class Shader
|
|
glUniformMatrix4fv(unifViewMatrix, false, buffer);
|
|
glUniformMatrix4fv(unifViewMatrix, false, buffer);
|
|
}
|
|
}
|
|
|
|
|
|
- protected static void setViewPort(int width, int height)
|
|
|
|
|
|
+ protected void setViewPort(float width, float height)
|
|
{
|
|
{
|
|
scale = 1;
|
|
scale = 1;
|
|
while(width / (scale + 1) >= 400 && height / (scale + 1) >= 300)
|
|
while(width / (scale + 1) >= 400 && height / (scale + 1) >= 300)
|
|
{
|
|
{
|
|
scale++;
|
|
scale++;
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ fontRenderer.setScale(scale);
|
|
|
|
|
|
- Shader.width = width / scale;
|
|
|
|
- Shader.height = height / scale;
|
|
|
|
|
|
+ this.width = width / scale;
|
|
|
|
+ this.height = height / scale;
|
|
|
|
|
|
updateViewMatrix();
|
|
updateViewMatrix();
|
|
}
|
|
}
|
|
|
|
|
|
- public static int getViewScale()
|
|
|
|
|
|
+ public int getViewScale()
|
|
{
|
|
{
|
|
return scale;
|
|
return scale;
|
|
}
|
|
}
|
|
|
|
|
|
- public static int getViewWidth()
|
|
|
|
|
|
+ public float getViewWidth()
|
|
{
|
|
{
|
|
return width;
|
|
return width;
|
|
}
|
|
}
|
|
|
|
|
|
- public static int getViewHeight()
|
|
|
|
|
|
+ public float getViewHeight()
|
|
{
|
|
{
|
|
return height;
|
|
return height;
|
|
}
|
|
}
|
|
|
|
|
|
- public static void updateMatrix()
|
|
|
|
|
|
+ public void updateMatrix()
|
|
{
|
|
{
|
|
glUniformMatrix4fv(unifModelMatrix, false, modelMatrix.getData());
|
|
glUniformMatrix4fv(unifModelMatrix, false, modelMatrix.getData());
|
|
}
|
|
}
|
|
|
|
|
|
- public static void pushMatrix()
|
|
|
|
|
|
+ public void pushMatrix()
|
|
{
|
|
{
|
|
modelMatrix.push();
|
|
modelMatrix.push();
|
|
}
|
|
}
|
|
|
|
|
|
- public static void popMatrix()
|
|
|
|
|
|
+ public void popMatrix()
|
|
{
|
|
{
|
|
modelMatrix.pop();
|
|
modelMatrix.pop();
|
|
}
|
|
}
|
|
|
|
|
|
- public static void translate(float tx, float ty)
|
|
|
|
|
|
+ public void translate(float tx, float ty)
|
|
{
|
|
{
|
|
modelMatrix.translate(tx, ty);
|
|
modelMatrix.translate(tx, ty);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void translateTo(float tx, float ty)
|
|
|
|
|
|
+ public void translateTo(float tx, float ty)
|
|
{
|
|
{
|
|
modelMatrix.translateTo(tx, ty);
|
|
modelMatrix.translateTo(tx, ty);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void scale(float sx, float sy)
|
|
|
|
|
|
+ public void scale(float sx, float sy)
|
|
{
|
|
{
|
|
modelMatrix.scale(sx, sy);
|
|
modelMatrix.scale(sx, sy);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void rotate(float angle)
|
|
|
|
|
|
+ public void rotate(float angle)
|
|
{
|
|
{
|
|
modelMatrix.rotate(angle);
|
|
modelMatrix.rotate(angle);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setAmbientLight(float r, float g, float b)
|
|
|
|
|
|
+ public void setAmbientLight(float r, float g, float b)
|
|
{
|
|
{
|
|
glUniform3f(unifAmbientLight, r, g, b);
|
|
glUniform3f(unifAmbientLight, r, g, b);
|
|
}
|
|
}
|
|
|
|
|
|
- private static void checkLightIndex(int index)
|
|
|
|
|
|
+ private void checkLightIndex(int index)
|
|
{
|
|
{
|
|
if(index < 0 || index > unifLight.length)
|
|
if(index < 0 || index > unifLight.length)
|
|
{
|
|
{
|
|
- throw new ShaderException("'" + index + "' is not a valid light index");
|
|
|
|
|
|
+ throw new ShaderException("'%d' is not a valid light index", index);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setLightColor(int index, float r, float g, float b)
|
|
|
|
|
|
+ public void setLightColor(int index, float r, float g, float b)
|
|
{
|
|
{
|
|
checkLightIndex(index);
|
|
checkLightIndex(index);
|
|
glUniform3f(unifLight[index][0], r, g, b);
|
|
glUniform3f(unifLight[index][0], r, g, b);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setLightLocation(int index, float x, float y)
|
|
|
|
|
|
+ public void setLightLocation(int index, float x, float y)
|
|
{
|
|
{
|
|
checkLightIndex(index);
|
|
checkLightIndex(index);
|
|
glUniform2f(unifLight[index][1], x, y);
|
|
glUniform2f(unifLight[index][1], x, y);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setLightStrength(int index, float strength)
|
|
|
|
|
|
+ public void setLightStrength(int index, float strength)
|
|
{
|
|
{
|
|
checkLightIndex(index);
|
|
checkLightIndex(index);
|
|
glUniform1f(unifLight[index][2], strength);
|
|
glUniform1f(unifLight[index][2], strength);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setTextureEnabled(boolean use)
|
|
|
|
|
|
+ public void setTextureEnabled(boolean use)
|
|
{
|
|
{
|
|
glUniform1i(unifUseTexture, use ? 1 : 0);
|
|
glUniform1i(unifUseTexture, use ? 1 : 0);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setColorEnabled(boolean use)
|
|
|
|
|
|
+ public void setColorEnabled(boolean use)
|
|
{
|
|
{
|
|
glUniform1i(unifUseColor, use ? 1 : 0);
|
|
glUniform1i(unifUseColor, use ? 1 : 0);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setMixColorEnabled(boolean use)
|
|
|
|
|
|
+ public void setMixColorEnabled(boolean use)
|
|
{
|
|
{
|
|
glUniform1i(unifUseMixColor, use ? 1 : 0);
|
|
glUniform1i(unifUseMixColor, use ? 1 : 0);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setMixColor(float r, float g, float b, float a)
|
|
|
|
|
|
+ public void setMixColor(float r, float g, float b, float a)
|
|
{
|
|
{
|
|
unifMixColor[0] = r;
|
|
unifMixColor[0] = r;
|
|
unifMixColor[1] = g;
|
|
unifMixColor[1] = g;
|
|
@@ -272,12 +244,12 @@ public final class Shader
|
|
glUniform4fv(unifMixColorLoc, unifMixColor);
|
|
glUniform4fv(unifMixColorLoc, unifMixColor);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setLightEnabled(boolean use)
|
|
|
|
|
|
+ public void setLightEnabled(boolean use)
|
|
{
|
|
{
|
|
glUniform1i(unifUseLight, use ? 1 : 0);
|
|
glUniform1i(unifUseLight, use ? 1 : 0);
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setDepthTestEnabled(boolean use)
|
|
|
|
|
|
+ public void setDepthTestEnabled(boolean use)
|
|
{
|
|
{
|
|
if(use)
|
|
if(use)
|
|
{
|
|
{
|
|
@@ -290,7 +262,7 @@ public final class Shader
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- public static void setBlendingEnabled(boolean use)
|
|
|
|
|
|
+ public void setBlendingEnabled(boolean use)
|
|
{
|
|
{
|
|
if(use)
|
|
if(use)
|
|
{
|
|
{
|
|
@@ -308,12 +280,12 @@ public final class Shader
|
|
// general stuff
|
|
// general stuff
|
|
// -------------------------------------------------------------------------
|
|
// -------------------------------------------------------------------------
|
|
|
|
|
|
- private static String[] readFile(String name)
|
|
|
|
|
|
+ private String[] readFile(String name)
|
|
{
|
|
{
|
|
- URL url = Shader.class.getClassLoader().getResource("me/hammerle/snuviengine/shader/" + name);
|
|
|
|
|
|
+ URL url = Renderer.class.getClassLoader().getResource("me/hammerle/snuviengine/shader/" + name);
|
|
if(url == null)
|
|
if(url == null)
|
|
{
|
|
{
|
|
- throw new ShaderException("failed reading shader '" + name + "'");
|
|
|
|
|
|
+ throw new ShaderException("failed reading shader '%s'", name);
|
|
}
|
|
}
|
|
ArrayList<String> strings = new ArrayList<>();
|
|
ArrayList<String> strings = new ArrayList<>();
|
|
try(Scanner scanner = new Scanner(url.openStream()))
|
|
try(Scanner scanner = new Scanner(url.openStream()))
|
|
@@ -326,11 +298,11 @@ public final class Shader
|
|
}
|
|
}
|
|
catch(IOException ex)
|
|
catch(IOException ex)
|
|
{
|
|
{
|
|
- throw new ShaderException("failed reading shader '" + name + "'");
|
|
|
|
|
|
+ throw new ShaderException("failed reading shader '%s'", name);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- private static String getError()
|
|
|
|
|
|
+ private String getError()
|
|
{
|
|
{
|
|
StringBuilder sb = new StringBuilder();
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
|
|
@@ -345,12 +317,12 @@ public final class Shader
|
|
return sb.toString();
|
|
return sb.toString();
|
|
}
|
|
}
|
|
|
|
|
|
- private static int createShaderProgram(String vertex, String fragment)
|
|
|
|
|
|
+ private int createShaderProgram(String vertex, String fragment)
|
|
{
|
|
{
|
|
// ---------------------------------------------------------------------
|
|
// ---------------------------------------------------------------------
|
|
// vertex shader
|
|
// vertex shader
|
|
// ---------------------------------------------------------------------
|
|
// ---------------------------------------------------------------------
|
|
-
|
|
|
|
|
|
+
|
|
String vShaderSource[] = readFile(vertex);
|
|
String vShaderSource[] = readFile(vertex);
|
|
|
|
|
|
int vShader = glCreateShader(GL_VERTEX_SHADER);
|
|
int vShader = glCreateShader(GL_VERTEX_SHADER);
|
|
@@ -360,13 +332,13 @@ public final class Shader
|
|
String error = getError();
|
|
String error = getError();
|
|
if(!error.isEmpty())
|
|
if(!error.isEmpty())
|
|
{
|
|
{
|
|
- throw new ShaderException("failed compiling vertex shader '" + vertex + "' " + error);
|
|
|
|
|
|
+ throw new ShaderException("failed compiling vertex shader '%s' %s", vertex, error);
|
|
}
|
|
}
|
|
|
|
|
|
int compiled = glGetShaderi(vShader, GL_COMPILE_STATUS);
|
|
int compiled = glGetShaderi(vShader, GL_COMPILE_STATUS);
|
|
if(compiled != 1)
|
|
if(compiled != 1)
|
|
{
|
|
{
|
|
- throw new ShaderException("failed compiling vertex shader '" + vertex + "' " + compiled + " " + glGetShaderInfoLog(vShader));
|
|
|
|
|
|
+ throw new ShaderException("failed compiling vertex shader '%s' %d %s", vertex, compiled, glGetShaderInfoLog(vShader));
|
|
}
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------
|
|
// ---------------------------------------------------------------------
|
|
@@ -382,13 +354,13 @@ public final class Shader
|
|
error = getError();
|
|
error = getError();
|
|
if(!error.isEmpty())
|
|
if(!error.isEmpty())
|
|
{
|
|
{
|
|
- throw new ShaderException("failed compiling fragment shader '" + fragment + "' " + error);
|
|
|
|
|
|
+ throw new ShaderException("failed compiling fragment shader '%s' %s", fragment, error);
|
|
}
|
|
}
|
|
|
|
|
|
compiled = glGetShaderi(fShader, GL_COMPILE_STATUS);
|
|
compiled = glGetShaderi(fShader, GL_COMPILE_STATUS);
|
|
if(compiled != 1)
|
|
if(compiled != 1)
|
|
{
|
|
{
|
|
- throw new ShaderException("failed compiling fragment shader '" + fragment + "' " + compiled + " " + glGetShaderInfoLog(fShader));
|
|
|
|
|
|
+ throw new ShaderException("failed compiling fragment shader '%s' %d %s", fragment, compiled, glGetShaderInfoLog(fShader));
|
|
}
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------
|
|
// ---------------------------------------------------------------------
|
|
@@ -403,13 +375,13 @@ public final class Shader
|
|
error = getError();
|
|
error = getError();
|
|
if(!error.isEmpty())
|
|
if(!error.isEmpty())
|
|
{
|
|
{
|
|
- throw new ShaderException("failed linking shaders '" + vertex + "' and '" + fragment + "' " + error);
|
|
|
|
|
|
+ throw new ShaderException("failed linking shaders '%s' and '%s' %s", vertex, fragment, error);
|
|
}
|
|
}
|
|
|
|
|
|
compiled = glGetProgrami(vfprogram, GL_LINK_STATUS);
|
|
compiled = glGetProgrami(vfprogram, GL_LINK_STATUS);
|
|
if(compiled != 1)
|
|
if(compiled != 1)
|
|
{
|
|
{
|
|
- throw new ShaderException("failed linking shaders '" + vertex + "' and '" + fragment + "' " + compiled + " " + glGetProgramInfoLog(vfprogram));
|
|
|
|
|
|
+ throw new ShaderException("failed linking shaders '%s' and '%s' %d %s", vertex, fragment, compiled, glGetProgramInfoLog(vfprogram));
|
|
}
|
|
}
|
|
|
|
|
|
glDeleteShader(vShader);
|
|
glDeleteShader(vShader);
|