Home Resume Blog
Adapting Our WebGL Point to WebGL2: Key Differences and Code Updates

In the last post, we created a simple WebGL program to draw a red point on a canvas, walking through the rendering pipeline and setting up basic shaders. That example used WebGL (version 1.0), which is widely supported but somewhat limited compared to its successor, WebGL2 (based on OpenGL ES 3.0). Since this series will focus on WebGL2, this post will explore the differences between WebGL and WebGL2 and adapt our previous code to run in a WebGL2 context.

Understanding WebGL vs. WebGL2

WebGL is based on OpenGL ES 2.0, while WebGL2 builds on OpenGL ES 3.0. This upgrade brings new features like 3D textures, multiple render targets, and enhanced shader capabilities, along with some syntax changes. For our simple example of drawing a single point with hardcoded coordinates, most of these advanced features won’t come into play yet, but we still need to adjust how we access the rendering context and write our shaders.

Updating the WebGL Context

In the previous post, we obtained the WebGL context with:

const gl = canvas.getContext("webgl");

For WebGL2, we simply change the argument to webgl2:

const gl = canvas.getContext("webgl2");

This gives us a WebGL2RenderingContext object instead of a WebGLRenderingContext. The new context supports all WebGL functionality plus WebGL2’s additional features.

Updating the Shaders

WebGL uses GLSL ES 1.00, while WebGL2 uses GLSL ES 3.00. This requires us to update our shader code with the correct version directive and adjust the fragment shader’s output syntax. Here’s how our original shaders evolve.

Original Vertex Shader (WebGL)
void main() {
  gl_Position = vec4(0.0, 0.0, 0.0, 1.0);   // Point at the center
  gl_PointSize = 100.0;                     // 100x100 pixel point
}
Updated Vertex Shader (WebGL2)
#version 300 es
void main() {
  gl_Position = vec4(0.0, 0.0, 0.0, 1.0);   // Point at the center
  gl_PointSize = 100.0;                     // 100x100 pixel point
}

The only change here is adding #version 300 es at the top to specify GLSL ES 3.00. Since our vertex shader hardcodes the point’s position and size without using attributes, no further adjustments are needed for this example. Also precision qualifiers are not required, defaulting to highp for floats.

Original Fragment Shader (WebGL)
void main() {
  gl_FragColor = vec4(1.0, 0.0, 0.0, 1.0);  // Red color
}
Updated Fragment Shader (WebGL2)
#version 300 es
precision highp float;
out vec4 fragColor;
void main() {
  fragColor = vec4(1.0, 0.0, 0.0, 1.0);  // Red color
}

The original uses gl_FragColor, which is deprecated in GLSL 3.00. Instead, webgl2 requires declaring an output variable, such as out vec4 fragColor. Additionally, precision for floats must be specified, in this case precision mediump float;

Putting It All Together

The rest of the code remains the same. Here’s the complete WebGL2 version of our simple point-drawing program:

const canvas = document.getElementById("example-canvas");
const gl = canvas.getContext("webgl2");

// Vertex shader: defines the point's position and size
const vertexShaderSource = `
#version 300 es
void main() {
  gl_Position = vec4(0.0, 0.0, 0.0, 1.0);   // Point at the center
  gl_PointSize = 100.0;                     // 100x100 pixel point
}
`;

// Fragment shader: sets the point color to red
const fragmentShaderSource = `
#version 300 es
precision highp float;
out vec4 fragColor;
void main() {
  fragColor = vec4(1.0, 0.0, 0.0, 1.0);  // Red color
}
`;

// Create and compile the vertex shader
const vertexShader = gl.createShader(gl.VERTEX_SHADER);
gl.shaderSource(vertexShader, vertexShaderSource);
gl.compileShader(vertexShader);

// Create and compile the fragment shader
const fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);
gl.shaderSource(fragmentShader, fragmentShaderSource);
gl.compileShader(fragmentShader);

// Create a program, attach shaders, and link them
const program = gl.createProgram();
gl.attachShader(program, vertexShader);
gl.attachShader(program, fragmentShader);
gl.linkProgram(program);
gl.useProgram(program);

// Clear the canvas and draw the point
gl.clearColor(1.0, 1.0, 1.0, 1.0); // White background
gl.clear(gl.COLOR_BUFFER_BIT); // Clear the canvas
gl.drawArrays(gl.POINTS, 0, 1); // Draw one point