Skip to main content

Using shaders to apply color in WebGL

Applying color to the vertices

In WebGL objects are built using sets of vertices, each of which has a position and a color. By default, all other pixels' colors (and all its other attributes, including position) are computed using interpolation, automatically creating smooth gradients. Previously, our vertex shader didn't apply any specific colors to the vertices. Between this and the fragment shader assigning the fixed color of white to each pixel, the entire square was rendered as solid white.


Let's say we want to render a gradient in which each corner of the square is a different color: red, blue, green, and white. The first thing to do is to establish these colors for the four vertices. To do this, we first need to create an array of vertex colors, then store it into a WebGL buffer. We'll do that by adding the following code to our initBuffers() function:


function initBuffers(){

  ...

  const colors = [

    1.0, 1.0, 1.0, 1.0, // white

    1.0, 0.0, 0.0, 1.0, // red

    0.0, 1.0, 0.0, 1.0, // green

    0.0, 0.0, 1.0, 1.0, // blue

  ];


  const colorBuffer = gl.createBuffer();

  gl.bindBuffer(gl.ARRAY_BUFFER, colorBuffer);

  gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);


  return {

    position: positionBuffer,

    color: colorBuffer,

  };

}

This code starts by creating a JavaScript array containing four 4-value vectors, one for each vertex color. Then a new WebGL buffer is allocated to store these colors, and the array is converted into floats and stored into the buffer.


To use these colors, the vertex shader needs to be updated to pull the appropriate color from the color buffer:


  const vsSource = `

    attribute vec4 aVertexPosition;

    attribute vec4 aVertexColor;


    uniform mat4 uModelViewMatrix;

    uniform mat4 uProjectionMatrix;


    varying lowp vec4 vColor;


    void main(void) {

      gl_Position = uProjectionMatrix * uModelViewMatrix * aVertexPosition;

      vColor = aVertexColor;

    }

  `;

The key difference here is that for each vertex, we pass its color using a varying to the fragment shader.


Coloring the fragments

As a refresher, here's what our fragment shader looked like previously:


  const fsSource = `

    void main() {

      gl_FragColor = vec4(1.0, 1.0, 1.0, 1.0);

    }

  `;

In order to pick up the interpolated color for each pixel, we need to change this to fetch the value from the vColor varying:


  const fsSource = `

    varying lowp vec4 vColor;


    void main(void) {

      gl_FragColor = vColor;

    }

  `;

Each fragment receives the interpolated color based on its position relative to the vertex positions instead of a fixed value.


Drawing using the colors

Next, it's necessary to add code to look up the attribute location for the colors and setup that attribute for the shader program:


  const programInfo = {

    program: shaderProgram,

    attribLocations: {

      vertexPosition: gl.getAttribLocation(shaderProgram, 'aVertexPosition'),

      vertexColor: gl.getAttribLocation(shaderProgram, 'aVertexColor'),

    },

    ...

Then, drawScene() can have the following added to it so it actually uses these colors when drawing the square:


  // Tell WebGL how to pull out the colors from the color buffer

  // into the vertexColor attribute.

  {

    const numComponents = 4;

    const type = gl.FLOAT;

    const normalize = false;

    const stride = 0;

    const offset = 0;

    gl.bindBuffer(gl.ARRAY_BUFFER, buffers.color);

    gl.vertexAttribPointer(

        programInfo.attribLocations.vertexColor,

        numComponents,

        type,

        normalize,

        stride,

        offset);

    gl.enableVertexAttribArray(

        programInfo.attribLocations.vertexColor);

  }

<meta charset='utf-8'><img src=""/>


Comments

Popular posts from this blog

Using textures in WebGL

Loading textures The first thing to do is add code to load the textures. In our case, we'll be using a single texture, mapped onto all six sides of our rotating cube, but the same technique can be used for any number of textures. Note: It's important to note that the loading of textures follows cross-domain rules; that is, you can only load textures from sites for which your content has CORS approval. See Cross-domain textures below for details. The code that loads the texture looks like this: // // Initialize a texture and load an image. // When the image finished loading copy it into the texture. // function loadTexture(gl, url) {   const texture = gl.createTexture();   gl.bindTexture(gl.TEXTURE_2D, texture);   // Because images have to be download over the internet   // they might take a moment until they are ready.   // Until then put a single pixel in the texture so we can   // use it immediately. When the image has finished downloading   // we'l...

Matrix math for the web

Matrices can be used to represent transformations of objects in space, and are an important tool to use in visualizations on the Web. This article explores how to create matrices and use them with  CSS3 transforms  and the  matrix3d  transform type. While this article uses CSS3 for the ease of explanations, matrices are a core concept used by many different technologies including WebGL and shaders. This article is also available as an  MDN content kit . The live examples use a collection of  utility functions  availabile under a global object named "MDN". What is a transformation matrix? There are many types of matrices, but the ones we are interested in are the 3D transformation matrices. These matrices consist of a set of 16 values arranged in a 4x4 grid. In JavaScript, it is easy to represent a matrix as an array. The typical starting point is to show the identity matrix. When this matrix is multiplied against another point or matrix then the result...

WebGL: 2D and 3D graphics for the web

WebGL (Web Graphics Library) is a JavaScript API for rendering high-performance interactive 3D and 2D graphics within any compatible web browser without the use of plug-ins. WebGL does so by introducing an API that closely conforms to OpenGL ES 2.0 that can be used in HTML5 <canvas> elements. This conformance makes it possible for the API to take advantage of hardware graphics acceleration provided by the user's device. Support for WebGL is present in Firefox 4+, Google Chrome 9+, Opera 12+, Safari 5.1+, Internet Explorer 11+, and Microsoft Edge build 10240+; however, the user's device must also have hardware that supports these features. The WebGL 2 API introduces support for much of the OpenGL ES 3.0 feature set; it's provided through the WebGL2RenderingContext interface. The <canvas> element is also used by the Canvas API to do 2D graphics on web pages.