-
Notifications
You must be signed in to change notification settings - Fork 186
ShaderLesson1
Preface
This series relies on the minimal lwjgl-basics API for shader and rendering utilities. However, the concepts should be universal enough that they could be applied to LibGDX, GLSL Sandbox, Love2D, iOS, or any other platforms that support GLSL. I'd recommend starting with the Textures tutorial as it covers essential concepts like filters and texture coordinates.
As discussed, we need to write vertex and fragment scripts in order for our shader program to work. Our first shaders will be very basic, and simply output the color red (ignoring the texture).
Note: In this series, we will use text files (.vert
and .frag
) for easier editing. When you go to release and distribute your games, you may want to embed the GLSL in your Java source as a String. Eclipse includes a feature for pasting multi-line strings which will be helpful.
Follow along with the full source code here.
Below is our setup code:
//load our shader program and sprite batch
try {
//read the files into strings
final String VERTEX = Util.readFile(Util.getResourceAsStream("res/shadertut/lesson1.vert"));
final String FRAGMENT = Util.readFile(Util.getResourceAsStream("res/shadertut/lesson1.frag"));
//create our shader program -- be sure to pass SpriteBatch's default attributes!
ShaderProgram program = new ShaderProgram(VERTEX, FRAGMENT, SpriteBatch.ATTRIBUTES);
//create our sprite batch
batch = new SpriteBatch(program);
} catch (Exception e) {
// ... handle the exception ...
}
For convenience, we use the Util class to read our text files.
We then create our shader program and specify the attribute locations with the third parameter. This tells ShaderProgram how the attributes will be laid out; since SpriteBatch expects them to be in a specific order (i.e. Position is expected at index 0).
Then, we create our SpriteBatch using our custom shader. Now we can render our sprites as per usual, and they will appear as red boxes:
protected void render() throws LWJGLException {
super.render();
// start our batch
batch.begin();
// draw some sprites... they will all be affected by our shaders
batch.draw(tex, 10, 10);
batch.draw(tex, 10, 320, 32, 32);
// end our batch
batch.end();
}
Let's take a look at what is going on. Here is the vertex shader, which works on every vertex that SpriteBatch sends to GL:
//incoming Position attribute from our SpriteBatch
attribute vec2 Position;
//the transformation matrix of our SpriteBatch
uniform mat4 u_projView;
void main() {
//transform our 2D screen space position into 3D world space
gl_Position = u_projView * vec4(Position.xy, 0.0, 1.0);
}
We simply take the Position attribute (given to us by our sprite batcher) -- such as (10, 10)
-- and transform it into 3D world-space coordinates that OpenGL can work with.
The fragment shader, which works on every fragment (or "pixel") that exists within our shape:
void main() {
//final color
gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);
}
Our fragment shader is also pretty simple -- it simply returns opaque red as the fragment (or "pixel") color.
attribute vec2 Position;
Seeing the above line, you may be wondering what exactly an attribute
is in GLSL. Think back to our brick sprite in the Textures tutorial:
As we explained in the Textures tutorial, we need to give OpenGL four vertices to make up our quad. Each vertex contains a number of attributes, such as Position
and TexCoord
:
In our case, we are ignoring the TexCoord
attribute since we don't need it. Instead, we only define the Position
attribute, using a vec2
(2-component float vector) as the data type to represent (x, y)
. SpriteBatch expects the name and data type to match accordingly.
Attributes can only be declared in vertex shaders. Also, attributes are read-only since they are passed from SpriteBatch. So we cannot assign them a value in GLSL.
The next line in our vertex shader brings us to another topic, uniforms:
uniform mat4 u_projView;
A uniform is like a script variable that we can set from Java. For example, if we needed to pass the mouse coordinates to a shader program, we would use a vec2
uniform and send the (x, y)
values to the shader every time the mouse moves. Like attributes, uniforms are read-only in the shader, so we cannot assign values to them in GLSL.
In our case, the vertex shader needs to transform the screen space coordinates from our SpriteBatch into 3D world-space coordinates. We do this by multiplying our Position
attribute by the combined transformation matrix of our SpriteBatch, which is named u_projView
(or SpriteBatch.U_PROJ_VIEW
). This leads to a 2D orthographic projection, where origin (0, 0)
is at the top left. e.g. (32, 7)
would be 32 pixels right, 7 pixels down.
gl_Position = u_projView * vec4(Position.xy, 0.0, 1.0);
Whenever we change the transformation matrix of our SpriteBatch (for example, when we call SpriteBatch.resize
), it will update the u_projView
uniform in our shader. It expects the name and data type to match; notice we are using mat4
as the GLSL data type to represent a 4x4 matrix.
A fragment shader usually ends by assigning a value to gl_FragColor
, which can be thought of in simple terms as a "return statement." This fragment shader is called on every pixel within our shape (in the case of SpriteBatch, rectangles), and for every fragment we are returning the color red. Thus, we end up with red boxes... Pretty simple.
gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);