OpenGL

Painting an Image

The main idea of this article is to build on top of the previous articles and paint an image, also to be able to move it around, scale it, and rotate it. But first we need to explain some basic topics that we will be using. Let us start with the shaders:

What are shaders

Shaders are little programs running in the GPU. They process inputs and generate outputs that can be used by other shader programs.

Shaders are written on a graphics programing language called GLSL (OpenGL Shading Language) that is similar to the C language. Each shader has a main function (the entry point) where the inputs are processed and any outputs are calculated.

In the previous article we saw an example of a vertex shader and a fragment shader they all look similar but they run in different stages of the graphics pipeline.

The next one is a simple example of a vertex shader:

#version 330 core
layout (location = 0) in vec3 a_pos; // the position variable has attribute position 0

out vec4 vertex_color; // specify a color output to the fragment shader

void main()
{
    gl_Position = vec4(a_pos, 1.0); // see how we directly give a vec3 to vec4's constructor
    vertex_color = vec4(0.5, 0.0, 0.0, 1.0); // set the output variable to a dark-red color
}

First we specify the OpenGL version (v3.3 core) that it supports. Then we declare an input for the position a_pos of type vec3 and we specify the attribute location (this will be used in the program to locate the attribute). We also declare a vertex color output vertex_color that we can set. By setting this output, it will be available to the next shaders in the pipeline to do any computation. The last thing that we see here is that we set the gl_Position variable. This is a special variable that OpenGL uses to set the position of a vertex.

The following is an example of the fragment shader:

#version 330 core
out vec4 FragColor;

in vec4 vertex_color; // the input variable from the vertex shader (same name and same type)

void main()
{
    FragColor = vertex_color;
}

The fragment shader’s main job is to set the pixel color that will be displayed. We only need to set FragColor which is a special output that OpenGL will use to know what is the color of the following output. In theory it is a bit more complex than that, but for now this should be fine to continue.

Shaders Inputs

There are 3 main types of inputs a shader can accept, and they are used for different purposes. But in general, inputs are used to pass information to a shader where they are used to calculate either vertex positions (vertex shader) or pixel color of a fragment (fragment shader).

Let us start with Attributes

Attributes

Attributes are set in our program code and are passed to the vertex shader. These are usually the values of the vertices positions, normals and texture coordinates.

We saw this one in the vertex shader above where we specify an attribute called a_pos which contains the information of the position of the vertices.

...
layout (location = 0) in vec3 a_pos;
...

Here we use the keyword layout plus the location of the attribute to tell OpenGL which slot this attribute is in. Here we also use the in keyword to specify that this variable is an input and finally we specify the type of the attribute. In this case we have a vector with 3 components for the vertex position in x, y and z.

Attibutes are set in code in this way:

glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 3 * sizeof(float), (void*)0);
glEnableVertexAttribArray(0);

We use the glVertexAttribPointer function to specify the layout of the vertex data. The first argument is the location of the attribute (same location as defined in the shader). Then we define the number of components of this attribute (a_pos is a vec3 so we need 3). Then comes the type of the data (floats) and whether the data should be normalized (GL_FALSE). Then we set the offset of each vertex data, in this case we only have positions in the array of data so the next vertex position would be 3 floats away, is distance (this is also called the stride). Finally we set the offset pointer to this attribute. Here we only have one attribute (vec3 a_pos) but this when we have more attributes packed inside a vertex array we need to make some calculation of where they are.

Finally, we enable the vertex attibute using the glEnableVertexAttribArray and passing the index or location specified in the previous step.

Squared Wave SVG