WebGL - Guía rápida

Hace unos años, las aplicaciones Java, como una combinación de applets y JOGL, se usaban para procesar gráficos 3D en la Web dirigiéndose a la GPU (Unidad de procesamiento gráfico). Como los subprogramas requieren una JVM para ejecutarse, se volvió difícil confiar en los subprogramas de Java. Unos años más tarde, la gente dejó de usar applets de Java.

Las API Stage3D proporcionadas por Adobe (Flash, AIR) ofrecían una arquitectura acelerada por hardware de GPU. Con estas tecnologías, los programadores podrían desarrollar aplicaciones con capacidades 2D y 3D en navegadores web, así como en plataformas IOS y Android. Dado que Flash era un software propietario, no se utilizó como estándar web.

En marzo de 2011, se lanzó WebGL. Es un software abierto que puede ejecutarse sin una JVM. Está completamente controlado por el navegador web.

La nueva versión de HTML 5 tiene varias características para admitir gráficos 3D como 2D Canvas, WebGL, SVG, transformaciones 3D CSS y SMIL. En este tutorial, cubriremos los conceptos básicos de WebGL.

¿Qué es OpenGL?

OpenGL (Open Graphics Library) es una API multiplataforma y en varios idiomas para gráficos 2D y 3D. Es una colección de comandos. OpenGL4.5 es la última versión de OpenGL. La siguiente tabla enumera un conjunto de tecnologías relacionadas con OpenGL.

API Tecnología utilizada
OpenGL ES Es la biblioteca para gráficos 2D y 3D en sistemas integrados, incluidas consolas, teléfonos, electrodomésticos y vehículos. OpenGL ES 3.1 es su última versión. Es mantenido por el Grupo Khronos www.khronos.org
JOGL Es el enlace de Java para OpenGL. JOGL 4.5 es su última versión y es mantenido por jogamp.org.
WebGL Es el enlace de JavaScript para OpenGL. WebGL 1.0 es su última versión y es mantenido por el grupo khronos .
OpenGLSL OpenGL Shading Language. Es un lenguaje de programación que acompaña a OpenGL 2.0 y superior. Es parte de la especificación principal de OpenGL 4.4. Es una API diseñada específicamente para sistemas integrados, como los presentes en teléfonos móviles y tabletas.

Note - En WebGL, usamos GLSL para escribir sombreadores.

¿Qué es WebGL?

WebGL (Web Graphics Library) es el nuevo estándar para gráficos 3D en la Web. Está diseñado con el propósito de representar gráficos 2D y gráficos 3D interactivos. Se deriva de la biblioteca ES 2.0 de OpenGL, que es una API 3D de bajo nivel para teléfonos y otros dispositivos móviles. WebGL proporciona una funcionalidad similar a ES 2.0 (sistemas integrados) y funciona bien en hardware de gráficos 3D moderno.

Es una API de JavaScript que se puede utilizar con HTML5. El código WebGL se escribe dentro de la etiqueta <canvas> de HTML5. Es una especificación que permite a los navegadores de Internet acceder a unidades de procesamiento gráfico (GPU) en aquellas computadoras donde se utilizaron.

Quién desarrolló WebGL

Un ingeniero de software estadounidense-serbio llamado Vladimir Vukicevic hizo el trabajo de base y dirigió la creación de WebGL

  • En 2007, Vladimir comenzó a trabajar en un OpenGL prototipo del elemento Canvas del documento HTML.

  • En marzo de 2011, Kronos Group creó WebGL.

Representación

La renderización es el proceso de generar una imagen a partir de un modelo utilizando programas de computadora. En gráficos, una escena virtual se describe utilizando información como geometría, punto de vista, textura, iluminación y sombreado, que se pasa a través de un programa de renderizado. El resultado de este programa de renderizado será una imagen digital.

Hay dos tipos de renderizado:

  • Software Rendering - Todos los cálculos de renderizado se realizan con la ayuda de la CPU.

  • Hardware Rendering - Todos los cálculos gráficos los realiza la GPU (unidad de procesamiento gráfico).

El renderizado se puede realizar de forma local o remota. Si la imagen que se va a renderizar es demasiado compleja, entonces el renderizado se realiza de forma remota en un servidor dedicado que tenga suficientes recursos de hardware necesarios para renderizar escenas complejas. También se llama comoserver-based rendering. La CPU también puede realizar el renderizado localmente. Se llama comoclient-based rendering.

WebGL sigue un enfoque de renderizado basado en el cliente para renderizar escenas 3D. Todo el procesamiento necesario para obtener una imagen se realiza de forma local utilizando el hardware gráfico del cliente.

GPU

Según NVIDIA, una GPU es "un procesador de un solo chip con transformación integrada, iluminación, configuración / recorte de triángulos y motores de renderizado capaces de procesar un mínimo de 10 millones de polígonos por segundo". A diferencia de los procesadores de varios núcleos con unos pocos núcleos optimizados para el procesamiento secuencial, una GPU consta de miles de núcleos más pequeños que procesan cargas de trabajo paralelas de manera eficiente. Por lo tanto, la GPU acelera la creación de imágenes en un búfer de cuadros (una parte de la memoria RAM que contiene datos de cuadros completos) destinados a la salida a una pantalla.

Computación acelerada por GPU

En la computación acelerada por GPU, la aplicación se carga en la CPU. Siempre que encuentre uncompute-intensiveparte del código, esa parte del código se cargará y ejecutará en la GPU. Le da al sistema la capacidad de procesar gráficos de manera eficiente.

La GPU tendrá una memoria separada y ejecutará múltiples copias de una pequeña parte del código a la vez. La GPU procesa todos los datos que están en su memoria local, no en la memoria central. Por lo tanto, los datos que la GPU necesita para procesar deben cargarse / copiarse en la memoria de la GPU y luego procesarse.

En los sistemas que tienen la arquitectura anterior, la sobrecarga de comunicación entre la CPU y la GPU debe reducirse para lograr un procesamiento más rápido de los programas 3D. Para ello, tenemos que copiar todos los datos y mantenerlos en la GPU, en lugar de comunicarnos con la GPU repetidamente.

Navegadores compatibles

Las siguientes tablas muestran una lista de navegadores que admiten WebGL:

Navegadores web

Nombre del navegador Versión Apoyo
IInternet Eexplorador 11 y más Soporte completo
Google Chrome 39 y más Soporte completo
Safari 8 Soporte completo
Firefox 36 y más Soporte parcial
Ópera 27 y más Soporte parcial

Navegadores móviles

Nombre del navegador Versión Apoyo
Chrome para Android 42 Soporte parcial
Navegador de Android 40 Soporte parcial
IOS Safari 8.3 Soporte completo
mini Opera 8 No soporta
Navegador Blackberry 10 Soporte completo
IE móvil 10 Soporte parcial

Ventajas de WebGL

Estas son las ventajas de usar WebGL:

  • JavaScript programming- Las aplicaciones WebGL están escritas en JavaScript. Con estas aplicaciones, puede interactuar directamente con otros elementos del documento HTML. También puede utilizar otras bibliotecas de JavaScript (por ejemplo, JQuery) y tecnologías HTML para enriquecer la aplicación WebGL.

  • Increasing support with mobile browsers - WebGL también es compatible con navegadores móviles como iOS safari, navegador Android y Chrome para Android.

  • Open source- WebGL es de código abierto. Puede acceder al código fuente de la biblioteca y comprender cómo funciona y cómo se desarrolló.

  • No need for compilation- JavaScript es un componente mitad programación y mitad HTML. Para ejecutar este script, no es necesario compilar el archivo. En su lugar, puede abrir directamente el archivo utilizando cualquiera de los navegadores y comprobar el resultado. Dado que las aplicaciones WebGL se desarrollan utilizando JavaScript, no es necesario compilar también aplicaciones WebGL.

  • Automatic memory management- JavaScript admite la gestión automática de memoria. No es necesaria la asignación manual de memoria. WebGL hereda esta característica de JavaScript.

  • Easy to set up- Dado que WebGL está integrado en HTML 5, no es necesario realizar una configuración adicional. Para escribir una aplicación WebGL, todo lo que necesita es un editor de texto y un navegador web.

Configuración del entorno

No es necesario configurar un entorno diferente para WebGL. Los navegadores que admiten WebGL tienen su propia configuración incorporada para WebGL.

Para crear aplicaciones gráficas en la web, HTML-5 proporciona un amplio conjunto de características como 2D Canvas, WebGL, SVG, transformaciones 3D CSS y SMIL. Para escribir aplicaciones WebGL, usamos el elemento canvas existente de HTML-5. Este capítulo proporciona una descripción general del elemento de lienzo 2D HTML-5.

Lienzo HTML5

HTML-5 <canvas>proporciona una opción fácil y potente para dibujar gráficos utilizando JavaScript. Se puede usar para dibujar gráficos, hacer composiciones de fotos o hacer animaciones simples (y no tan simples).

Aquí hay un simple <canvas> elemento que tiene solo dos atributos específicos width y height además de todos los atributos básicos de HTML-5 como id, nombre y clase.

Sintaxis

La sintaxis de la etiqueta de lienzo HTML se proporciona a continuación. Debe mencionar el nombre del lienzo entre comillas dobles (“”).

<canvas id = "mycanvas" width = "100" height = "100"></canvas>

Atributos del lienzo

La etiqueta de lienzo tiene tres atributos, a saber, id, ancho y alto.

  • Id- Id representa el identificador del elemento canvas en el Modelo de objetos de documento (DOM) .

  • Width - Ancho representa el ancho del lienzo.

  • Height - Altura representa la altura del lienzo.

Estos atributos determinan el tamaño del lienzo. Si un programador no los especifica en la etiqueta de lienzo, los navegadores como Firefox, Chrome y Web Kit, de forma predeterminada, proporcionan un elemento de lienzo de tamaño 300 × 150.

Ejemplo: crear un lienzo

El siguiente código muestra cómo crear un lienzo. Hemos utilizado CSS para darle un borde de color al lienzo.

<html>
   <head>
      <style>
         #mycanvas{border:1px solid red;}
      </style>
   </head>
   <body>
      <canvas id = "mycanvas" width = "100" height = "100"></canvas>
   </body>
</html>

Al ejecutarse, el código anterior producirá la siguiente salida:

El contexto de renderizado

El <canvas> está inicialmente en blanco. Para mostrar algo en el elemento lienzo, tenemos que usar un lenguaje de secuencias de comandos. Este lenguaje de secuencias de comandos debe acceder al contexto de representación y basarse en él.

El elemento canvas tiene un método DOM llamado getContext(), que se utiliza para obtener el contexto de representación y sus funciones de dibujo. Este método toma un parámetro, el tipo de contexto2d.

El siguiente código debe escribirse para obtener el contexto requerido. Puede escribir este script dentro de la etiqueta del cuerpo como se muestra a continuación.

<!DOCTYPE HTML>
<html>
   <body>
      <canvas id = "mycanvas" width = "600" height = "200"></canvas>

      <script>
         var canvas = document.getElementById('mycanvas');
         var context = canvas.getContext('2d');
			
         context.font = '20pt Calibri';
         context.fillStyle = 'green';
         context.fillText('Welcome to Tutorialspoint', 70, 70);
      </script>
   </body>
</html>

Al ejecutarse, el código anterior producirá la siguiente salida:

Para obtener más ejemplos sobre HTML-5 2D Canvas, consulte el siguiente enlace HTML-5 Canvas .

Contexto de WebGL

HTML5 Canvas también se utiliza para escribir aplicaciones WebGL. Para crear un contexto de representación WebGL en el elemento canvas, debe pasar la cadenaexperimental-webgl, en vez de 2d al canvas.getContext()método. Algunos navegadores solo admiten 'webgl'.

<!DOCTYPE html>
<html>
   <canvas id = 'my_canvas'></canvas>
	
   <script>
      var canvas = document.getElementById('my_canvas');
      var gl = canvas.getContext('experimental-webgl');
      gl.clearColor(0.9,0.9,0.8,1);
      gl.clear(gl.COLOR_BUFFER_BIT);
   </script>
</html>

Al ejecutarse, el código anterior producirá la siguiente salida:

WebGL es principalmente una API de rasterización de bajo nivel en lugar de una API 3D. Para dibujar una imagen usando WebGL, debe pasar un vector que represente la imagen. Luego convierte el vector dado en formato de píxeles usando OpenGL SL y muestra la imagen en la pantalla. Escribir una aplicación WebGL implica una serie de pasos que explicaremos en este capítulo.

WebGL - Sistema de coordenadas

Al igual que cualquier otro sistema 3D, tendrá los ejes x, y y z en WebGL, donde z eje significa depth. Las coordenadas en WebGL están restringidas a (1, 1, 1) y (-1, -1, - 1). Significa que si considera que la pantalla que proyecta gráficos WebGL como un cubo, entonces una esquina del cubo será (1, 1, 1) y la esquina opuesta será (-1, -1, -1). WebGL no mostrará nada que se dibuje más allá de estos límites.

El siguiente diagrama muestra el sistema de coordenadas WebGL. El eje z significa profundidad. Un valor positivo de z indica que el objeto está cerca de la pantalla / espectador, mientras que un valor negativo de z indica que el objeto está lejos de la pantalla. Asimismo, un valor positivo de x indica que el objeto está en el lado derecho de la pantalla y un valor negativo indica que el objeto está en el lado izquierdo. Del mismo modo, los valores positivos y negativos de y indican si el objeto está en la parte superior o inferior de la pantalla.

Gráficos WebGL

Después de obtener el contexto WebGL del objeto de lienzo, puede comenzar a dibujar elementos gráficos utilizando la API WebGL en JavaScript.

A continuación, se incluyen algunos términos fundamentales que debe conocer antes de comenzar con WebGL.

Vértices

Generalmente, para dibujar objetos como un polígono, marcamos los puntos en el plano y los unimos para formar un polígono deseado. UNvertexes un punto que define la conjunción de los bordes de un objeto 3D. Está representado por tres valores de coma flotante, cada uno de los cuales representa los ejes x, y, z respectivamente.

Ejemplo

En el siguiente ejemplo, estamos dibujando un triángulo con los siguientes vértices: (0.5, 0.5), (-0.5, 0.5), (-0.5, -0.5).

Note - Tenemos que almacenar estos vértices manualmente usando matrices JavaScript y pasarlos a la canalización de renderizado de WebGL usando el búfer de vértices.

Índices

En WebGL, los valores numéricos se utilizan para identificar los vértices. Estos valores numéricos se conocen como índices. Estos índices se utilizan para dibujar mallas en WebGL.

Note - Al igual que los vértices, almacenamos los índices usando matrices de JavaScript y los pasamos a la canalización de renderizado WebGL usando el búfer de índice.

Matrices

A diferencia de OpenGL y JoGL, no hay métodos predefinidos en WebGL para representar los vértices directamente. Tenemos que almacenarlos manualmente usando matrices JavaScript.

Ejemplo

var vertices = [ 0.5, 0.5, 0.1,-0.5, 0.5,-0.5]

Amortiguadores

Los búferes son las áreas de memoria de WebGL que contienen los datos. Hay varios búferes, a saber, búfer de dibujo, búfer de cuadro, búfer vetex y búfer de índice. losvertex buffer y index buffer se utilizan para describir y procesar la geometría del modelo.

Los objetos de búfer de vértice almacenan datos sobre los vértices, mientras que los objetos de búfer de índice almacenan datos sobre los índices. Después de almacenar los vértices en matrices, los pasamos a la canalización de gráficos WegGL utilizando estos objetos Buffer.

Frame bufferes una parte de la memoria gráfica que contiene los datos de la escena. Este búfer contiene detalles como el ancho y alto de la superficie (en píxeles), el color de cada píxel, la profundidad y los búferes de la plantilla.

Malla

Para dibujar objetos 2D o 3D, la API de WebGL proporciona dos métodos, a saber, drawArrays() y drawElements(). Estos dos métodos aceptan un parámetro llamadomodecon el que puede seleccionar el objeto que desea dibujar. Las opciones proporcionadas por este campo están restringidas a puntos, líneas y triángulos.

Para dibujar un objeto 3D usando estos dos métodos, tenemos que construir uno o más polígonos primitivos usando puntos, líneas o triángulos. A partir de entonces, utilizando esos polígonos primitivos, podemos formar una malla.

Un objeto 3D dibujado con polígonos primitivos se llama mesh. WebGL ofrece varias formas de dibujar objetos gráficos en 3D, sin embargo, los usuarios normalmente prefieren dibujar una malla.

Ejemplo

En el siguiente ejemplo, puedes observar que hemos dibujado un cuadrado usando dos triángulos → {1, 2, 3} y {4, 1, 3}.

Programas de sombreado

Normalmente usamos triángulos para construir mallas. Dado que WebGL utiliza computación acelerada por GPU, la información sobre estos triángulos debe transferirse de la CPU a la GPU, lo que requiere una gran sobrecarga de comunicación.

WebGL proporciona una solución para reducir la sobrecarga de comunicación. Dado que utiliza ES SL (Embedded System Shader Language) que se ejecuta en GPU, escribimos todos los programas necesarios para dibujar elementos gráficos en el sistema cliente utilizandoshader programs (los programas que escribimos usando OpenGL ES Shading Language / GLSL).

Estos sombreadores son los programas para GPU y el lenguaje utilizado para escribir programas de sombreado es GLSL. En estos sombreadores, definimos exactamente cómo los vértices, las transformaciones, los materiales, las luces y la cámara interactúan entre sí para crear una imagen en particular.

En resumen, es un fragmento que implementa algoritmos para obtener píxeles para una malla. Discutiremos más sobre los sombreadores en capítulos posteriores. Hay dos tipos de sombreadores: Vertex Shader y Fragment Shader.

Sombreador de vértices

El sombreador de texto es el código de programa llamado en cada vértice. Se utiliza para transformar (mover) la geometría (ej .: triángulo) de un lugar a otro. Maneja los datos de cada vértice (datos por vértice) como coordenadas de vértice, normales, colores y coordenadas de textura.

En el ES GLcódigo del sombreador de vértices, los programadores tienen que definir atributos para manejar los datos. Estos atributos apuntan a unVertex Buffer Object escrito en JavaScript.

Las siguientes tareas se pueden realizar utilizando sombreadores de vértices:

  • Transformación de vértice
  • Normalización y transformación normal
  • Generación de coordenadas de textura
  • Transformación de coordenadas de textura
  • Lighting
  • Aplicación de material de color

Shader de fragmentos (Pixel Shader)

Una malla está formada por múltiples triángulos, y la superficie de cada uno de los triángulos se conoce como fragment. El sombreador de fragmentos es el código que se ejecuta en todos los píxeles de cada fragmento. Está escrito para calcular y rellenar el color en píxeles individuales .

Las siguientes tareas se pueden realizar con los sombreadores de fragmentos:

  • Operaciones sobre valores interpolados
  • Acceso a texturas
  • Aplicación de textura
  • Fog
  • Suma de colores

Variables OpenGL ES SL

La forma completa de OpenGL ES SLes OpenGL Embedded System Shading Language. Para manejar los datos en los programas de sombreado, ES SL proporciona tres tipos de variables. Son los siguientes:

  • Attributes- Estas variables contienen los valores de entrada del programa de sombreado de vértices. Los atributos apuntan a los objetos de búfer de vértice que contienen datos por vértice. Cada vez que se invoca el sombreador de vértices, los atributos apuntan a VBO de diferentes vértices.

  • Uniforms - Estas variables contienen los datos de entrada que son comunes para los sombreadores de vértices y fragmentos, como la posición de la luz, las coordenadas de textura y el color.

  • Varyings - Estas variables se utilizan para pasar los datos del sombreador de vértices al sombreador de fragmentos.

Con todos estos conceptos básicos, ahora pasaremos a discutir la canalización de gráficos.

Para renderizar gráficos en 3D, tenemos que seguir una secuencia de pasos. Estos pasos se conocen comographics pipeline o rendering pipeline. El siguiente diagrama muestra la canalización de gráficos WebGL.

En las siguientes secciones, discutiremos uno por uno el rol de cada paso en la tubería.

JavaScript

Mientras desarrollamos aplicaciones WebGL, escribimos código de lenguaje Shader para comunicarnos con la GPU. JavaScript se utiliza para escribir el código de control del programa, que incluye las siguientes acciones:

  • Initialize WebGL - JavaScript se utiliza para inicializar el contexto WebGL.

  • Create arrays - Creamos matrices JavaScript para contener los datos de la geometría.

  • Buffer objects - Creamos objetos de búfer (vértice e índice) pasando las matrices como parámetros.

  • Shaders - Creamos, compilamos y vinculamos los sombreadores usando JavaScript.

  • Attributes - Podemos crear atributos, habilitarlos y asociarlos con objetos de búfer usando JavaScript.

  • Uniforms - También podemos asociar los uniformes usando JavaScript.

  • Transformation matrix - Usando JavaScript, podemos crear una matriz de transformación.

Inicialmente creamos los datos para la geometría requerida y los pasamos a los sombreadores en forma de búferes. La variable de atributo del lenguaje de sombreado apunta a los objetos de búfer, que se pasan como entradas al sombreador de vértices.

Sombreador de vértices

Cuando comenzamos el proceso de renderizado invocando los métodos drawElements() y drawArray(), el sombreador de vértices se ejecuta para cada vértice proporcionado en el objeto de búfer de vértice. Calcula la posición de cada vértice de un polígono primitivo y lo almacena en la variantegl_position. También calcula los otros atributos comocolor, texture coordinatesy vertices que normalmente están asociados con un vértice.

Asamblea primitiva

Después de calcular la posición y otros detalles de cada vértice, la siguiente fase es la primitive assembly stage. Aquí los triángulos se ensamblan y se pasan al rasterizador.

Rasterización

En el paso de rasterización, se determinan los píxeles de la imagen final de la primitiva. Tiene dos pasos:

  • Culling- Inicialmente se determina la orientación (¿está mirando hacia adelante o hacia atrás?) Del polígono. Todos aquellos triángulos con una orientación incorrecta que no son visibles en el área de visualización se descartan. Este proceso se llama sacrificio.

  • Clipping- Si un triángulo está parcialmente fuera del área de visualización, se eliminará la parte fuera del área de visualización. Este proceso se conoce como recorte.

Shader de fragmentos

El sombreador de fragmentos obtiene

  • datos del sombreador de vértices en variables variables,
  • primitivas de la etapa de rasterización, y luego
  • calcula los valores de color para cada píxel entre los vértices.

El sombreador de fragmentos almacena los valores de color de cada píxel en cada fragmento. Se puede acceder a estos valores de color durante las operaciones de fragmentos, que discutiremos a continuación.

Operaciones de fragmentos

Las operaciones de fragmentos se llevan a cabo después de determinar el color de cada píxel en la primitiva. Estas operaciones de fragmentos pueden incluir lo siguiente:

  • Depth
  • Mezcla de tampón de color
  • Dithering

Una vez que se procesan todos los fragmentos, se forma una imagen 2D y se muestra en la pantalla. losframe buffer es el destino final de la canalización de renderizado.

Búfer de fotogramas

El búfer de fotogramas es una parte de la memoria gráfica que contiene los datos de la escena. Este búfer contiene detalles como el ancho y alto de la superficie (en píxeles), el color de cada píxel y los búferes de profundidad y plantilla.

Hemos discutido los conceptos básicos de WebGL y la canalización de WebGL (un procedimiento que se sigue para renderizar aplicaciones de gráficos). En este capítulo, vamos a tomar una aplicación de muestra para crear un triángulo usando WebGL y observar los pasos seguidos en la aplicación.

Estructura de la aplicación WebGL

El código de la aplicación WebGL es una combinación de JavaScript y OpenGL Shader Language.

  • Se requiere JavaScript para comunicarse con la CPU
  • Se requiere OpenGL Shader Language para comunicarse con la GPU.

Aplicación de muestra

Tomemos ahora un ejemplo simple para aprender a usar WebGL para dibujar un triángulo simple con coordenadas 2D.

<!doctype html>
<html>
   <body>
      <canvas width = "300" height = "300" id = "my_Canvas"></canvas>
		
      <script>
         /* Step1: Prepare the canvas and get WebGL context */

         var canvas = document.getElementById('my_Canvas');
         var gl = canvas.getContext('experimental-webgl');

         /* Step2: Define the geometry and store it in buffer objects */

         var vertices = [-0.5, 0.5, -0.5, -0.5, 0.0, -0.5,];

         // Create a new buffer object
         var vertex_buffer = gl.createBuffer();

         // Bind an empty array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         
         // Pass the vertices data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         /* Step3: Create and compile Shader programs */

         // Vertex shader source code
         var vertCode =
            'attribute vec2 coordinates;' + 
            'void main(void) {' + ' gl_Position = vec4(coordinates,0.0, 1.0);' + '}';

         //Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         //Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         //Compile the vertex shader
         gl.compileShader(vertShader);

         //Fragment shader source code
         var fragCode = 'void main(void) {' + 'gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' + '}';

         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragment shader
         gl.compileShader(fragShader);

         // Create a shader program object to store combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader); 
         
         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /* Step 4: Associate the shader programs to buffer objects */

         //Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         //Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         //point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 2, gl.FLOAT, false, 0, 0);

         //Enable the attribute
         gl.enableVertexAttribArray(coord);

         /* Step5: Drawing the required object (triangle) */

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST); 
         
         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawArrays(gl.TRIANGLES, 0, 3);
      </script>
   </body>
</html>

Producirá el siguiente resultado:

Si observa el programa anterior con atención, hemos seguido cinco pasos secuenciales para dibujar un triángulo simple usando WebGL. Los pasos son los siguientes:

Step 1 − Prepare the canvas and get WebGL rendering context

Obtenemos el objeto de lienzo HTML actual y obtenemos su contexto de representación WebGL.

Step 2 − Define the geometry and store it in buffer objects

Definimos los atributos de la geometría como vértices, índices, color, etc., y los almacenamos en las matrices de JavaScript. Luego, creamos uno o más objetos de búfer y pasamos las matrices que contienen los datos al objeto de búfer respectivo. En el ejemplo, almacenamos los vértices del triángulo en una matriz de JavaScript y pasamos esta matriz a un objeto de búfer de vértice.

Step 3 − Create and compile Shader programs

Escribimos programas de sombreado de vértices y de fragmentos, los compilamos y creamos un programa combinado vinculando estos dos programas.

Step 4 − Associate the shader programs with buffer objects

Asociamos los objetos de búfer y el programa de sombreado combinado.

Step 5 − Drawing the required object (triangle)

Este paso incluye operaciones como borrar el color, borrar el bit del búfer, habilitar la prueba de profundidad, configurar el puerto de visualización, etc. Finalmente, debe dibujar las primitivas requeridas utilizando uno de los métodos: drawArrays() o drawElements().

Todos estos pasos se explican con más detalle en este tutorial.

Para escribir una aplicación WebGL, el primer paso es obtener el objeto de contexto de representación de WebGL. Este objeto interactúa con el búfer de dibujo de WebGL y puede llamar a todos los métodos de WebGL. Se realizan las siguientes operaciones para obtener el contexto WebGL:

  • Crea un lienzo HTML-5
  • Obtener el ID de lienzo
  • Obtenga WebGL

Creación de un elemento Canvas HTML-5

En el Capítulo 5, discutimos cómo crear un elemento de lienzo HTML-5. Dentro del cuerpo del documento HTML-5, escriba un lienzo, asígnele un nombre y páselo como parámetro al ID de atributo. Puede definir las dimensiones del lienzo utilizando los atributos de ancho y alto (opcional).

Ejemplo

El siguiente ejemplo muestra cómo crear un elemento de lienzo con las dimensiones 500 × 500. Hemos creado un borde para el lienzo usando CSS para la visibilidad. Copie y pegue el siguiente código en un archivo con el nombremy_canvas.html.

<!DOCTYPE HTML>
<html>
   <head>
      <style>
         #mycanvas{border:1px solid blue;}
      </style>
   </head>
   <body>
      <canvas id = "mycanvas" width = "300" height = "300"></canvas>
   </body>
</html>

Producirá el siguiente resultado:

Obtener el ID de Canvas

Después de crear el lienzo, debe obtener el contexto de WebGL. Lo primero que debe hacer para obtener un contexto de dibujo de WebGL es obtener la identificación del elemento de lienzo actual.

Canvas ID se adquiere llamando al método DOM (Document Object Model) getElementById(). Este método acepta un valor de cadena como parámetro, por lo que le pasamos el nombre del lienzo actual.

Por ejemplo, si el nombre del lienzo es my_canvas, luego se obtiene el ID del lienzo como se muestra a continuación

var canvas = document.getElementById('my_Canvas');

Obtenga el contexto de dibujo de WebGL

Para obtener el objeto WebGLRenderingContext (o el objeto de contexto de dibujo de WebGL o simplemente el contexto de WebGL), llame al getContext() método de la corriente HTMLCanvasElement. La sintaxis de getContext () es la siguiente:

canvas.getContext(contextType, contextAttributes);

Pasa las cuerdas webgl o experimental-webgl como el contentType. loscontextAttributesel parámetro es opcional. (Mientras continúa con este paso, asegúrese de que su navegador implemente WebGL versión 1 (OpenGL ES 2.0)).

El siguiente fragmento de código muestra cómo obtener el contexto de representación de WebGL. aquígl es la variable de referencia al objeto de contexto obtenido.

var canvas = document.getElementById('my_Canvas');
var gl = canvas.getContext('experimental-webgl');

WebGLContextAttributes

El parámetro WebGLContextAttributesno es obligatorio. Este parámetro proporciona varias opciones que aceptan valores booleanos que se enumeran a continuación:

No Señor. Atributos y descripción
1

Alpha

Si su valor es verdadero, proporciona un búfer alfa al lienzo.

De forma predeterminada, su valor es verdadero.

2

depth

Si su valor es verdadero, obtendrá un búfer de dibujo que contiene un búfer de profundidad de al menos 16 bits.

De forma predeterminada, su valor es verdadero.

3

stencil

Si su valor es verdadero, obtendrá un búfer de dibujo que contiene un búfer de plantilla de al menos 8 bits.

Por defecto, su valor es falso.

4

antialias

Si su valor es verdadero, obtendrá un búfer de dibujo que realiza suavizado.

De forma predeterminada, su valor es verdadero.

5

premultipliedAlpha

Si su valor es verdadero, obtendrá un búfer de dibujo que contiene colores con alfa pre-multiplicado.

De forma predeterminada, su valor es verdadero.

6

preserveDrawingBuffer

Si su valor es verdadero, los búferes no se borrarán y conservarán sus valores hasta que el autor los borre o los sobrescriba.

Por defecto, su valor es falso.

El siguiente fragmento de código muestra cómo crear un contexto WebGL con un búfer de plantilla, que no funcionará anti-aliasing.

var canvas = document.getElementById('canvas1');
var context = canvas.getContext('webgl', { antialias: false, stencil: true });

En el momento de crear WebGLRenderingContext, se crea un búfer de dibujo. El objeto Context administra el estado de OpenGL y se procesa en el búfer de dibujo.

WebGLRenderingContext

Es la interfaz principal en WebGL. Representa el contexto de dibujo de WebGL. Esta interfaz contiene todos los métodos utilizados para realizar diversas tareas en el búfer de dibujo. Los atributos de esta interfaz se dan en la siguiente tabla.

No Señor. Atributos y descripción
1

Canvas

Esta es una referencia al elemento canvas que creó este contexto.

2

drawingBufferWidth

Este atributo representa el ancho real del búfer de dibujo. Puede diferir del atributo de ancho de HTMLCanvasElement.

3

drawingBufferHeight

Este atributo representa la altura real del búfer de dibujo. Puede diferir del atributo de altura de HTMLCanvasElement.

Después de obtener el contexto WebGL, debe definir la geometría de la primitiva (objeto que desea dibujar) y almacenarla. En WebGL, definimos los detalles de una geometría, por ejemplo, vértices, índices, color de la primitiva, utilizando matrices JavaScript. Para pasar estos detalles a los programas de sombreado, tenemos que crear los objetos de búfer y almacenar (adjuntar) las matrices de JavaScript que contienen los datos en los búferes respectivos.

Note: Posteriormente, estos objetos de búfer se asociarán con los atributos del programa de sombreado (sombreador de vértices).

Definición de la geometría requerida

Un modelo 2D o 3D dibujado con vértices se llama mesh. Cada faceta de una malla se llamapolygon y un polígono está formado por 3 o más vértices.

Para dibujar modelos en el contexto de representación de WebGL, debe definir los vértices y los índices utilizando matrices de JavaScript. Por ejemplo, si queremos crear un triángulo que se encuentre en las coordenadas {(5,5), (-5,5), (-5, -5)} como se muestra en el diagrama, entonces puede crear una matriz para los vértices como -

var vertices = [
   0.5,0.5,    //Vertex 1
   0.5,-0.5,   //Vertex 2
   -0.5,-0.5,  //Vertex 3
];

Del mismo modo, puede crear una matriz para los índices. Los índices para los índices triangulares anteriores serán [0, 1, 2] y se pueden definir como:

var indices = [ 0,1,2 ]

Para una mejor comprensión de los índices, considere modelos más complejos como el cuadrado. Podemos representar un cuadrado como un conjunto de dos triángulos. Si (0,3,1) y (3,1,2) son los dos triángulos con los que pretendemos dibujar un cuadrado, entonces los índices se definirán como -

var indices = [0,3,1,3,1,2];

Note -

Para dibujar primitivas, WebGL proporciona los siguientes dos métodos:

  • drawArrays() - Mientras usamos este método, pasamos los vértices de la primitiva usando matrices JavaScript.

  • drawElements() - Mientras usamos este método, pasamos tanto los vértices como los índices de la primitiva usando una matriz de JavaScript.

Objetos de búfer

Un objeto de búfer es un mecanismo proporcionado por WebGL que indica un área de memoria asignada en el sistema. En estos objetos de búfer, puede almacenar datos del modelo que desea dibujar, correspondientes a vértices, índices, color, etc.

Con estos objetos de búfer, puede pasar varios datos al programa de sombreado (sombreador de vértices) a través de una de sus variables de atributo. Dado que estos objetos de búfer residen en la memoria de la GPU, se pueden procesar directamente, lo que a su vez mejora el rendimiento.

Para procesar la geometría, existen dos tipos de objetos de zona de influencia. Ellos son -

  • Vertex buffer object (VBO)- Contiene los datos por vértice del modelo gráfico que se va a renderizar. Usamos objetos de búfer de vértice en WebGL para almacenar y procesar los datos relacionados con los vértices, como coordenadas de vértice, normales, colores y coordenadas de textura.

  • Index buffer objects (IBO) - Contiene los índices (datos de índice) del modelo gráfico que se va a renderizar.

Después de definir la geometría requerida y almacenarlas en matrices de JavaScript, debe pasar estas matrices a los objetos de búfer, desde donde los datos se pasarán a los programas de sombreado. Se deben seguir los siguientes pasos para almacenar datos en los búferes.

  • Crea un búfer vacío.

  • Vincula un objeto de matriz apropiado al búfer vacío.

  • Pase los datos (vértices / índices) al búfer usando uno de los typed arrays.

  • Desenlazar el búfer (opcional).

Crear un búfer

Para crear un objeto de búfer vacío, WebGL proporciona un método llamado createBuffer(). Este método devuelve un objeto de búfer recién creado, si la creación se realizó correctamente; de lo contrario, devuelve un valor nulo en caso de falla.

WebGL funciona como una máquina de estado. Una vez que se crea un búfer, cualquier operación de búfer posterior se ejecutará en el búfer actual hasta que lo desatemos. Utilice el siguiente código para crear un búfer:

var vertex_buffer = gl.createBuffer();

Note - gl es la variable de referencia al contexto WebGL actual.

Vincular el búfer

Después de crear un objeto de búfer vacío, debe vincularle un búfer de matriz adecuado (destino). WebGL proporciona un método llamadobindBuffer() para este propósito.

Sintaxis

La sintaxis de bindBuffer() El método es el siguiente:

void bindBuffer (enum target, Object buffer)

Este método acepta dos parámetros y se analizan a continuación.

target- La primera variable es un valor de enumeración que representa el tipo de búfer que queremos vincular al búfer vacío. Tiene dos valores de enumeración predefinidos como opciones para este parámetro. Ellos son -

  • ARRAY_BUFFER que representa datos de vértice.

  • ELEMENT_ARRAY_BUFFER que representa datos de índice.

Object buffer- El segundo es la variable de referencia al objeto de búfer creado en el paso anterior. La variable de referencia puede ser de un objeto de búfer de vértice o de un objeto de búfer de índice.

Ejemplo

El siguiente fragmento de código muestra cómo utilizar el método bindBuffer ().

//vertex buffer
var vertex_buffer = gl.createBuffer();
gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

//Index buffer
var Index_Buffer = gl.createBuffer();
gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);

Pasar datos al búfer

El siguiente paso es pasar los datos (vértices / índices) al búfer. Hasta ahora, los datos tienen la forma de una matriz y antes de pasarlos al búfer, debemos envolverlos en una de las matrices escritas en WebGL. WebGL proporciona un método llamadobufferData() para este propósito.

Sintaxis

La sintaxis del método bufferData () es la siguiente:

void bufferData (enum target, Object data, enum usage)

Este método acepta tres parámetros y se analizan a continuación:

target - El primer parámetro es un valor de enumeración que representa el tipo de búfer de matriz que usamos. Las opciones para este parámetro son -

  • ARRAY_BUFFER que representa vertex data.

  • ELEMENT_ARRAY_BUFFER que representa index data.

Object data- El segundo parámetro es el valor del objeto que contiene los datos que se escribirán en el objeto de búfer. Aquí tenemos que pasar los datos usandotyped arrays.

Usage- El tercer parámetro de este método es una variable de enumeración que especifica cómo utilizar los datos del objeto de búfer (datos almacenados) para dibujar formas. Hay tres opciones para este parámetro que se enumeran a continuación.

  • gl.STATIC_DRAW - Los datos se especificarán una vez y se utilizarán muchas veces.

  • gl.STREAM_DRAW - Los datos se especificarán una vez y se utilizarán varias veces.

  • gl.DYNAMIC_DRAW - Los datos se especificarán repetidamente y se utilizarán muchas veces.

Ejemplo

El siguiente fragmento de código muestra cómo utilizar el bufferData()método. Suponga que los vértices y los índices son las matrices que contienen los datos del vértice y del índice respectivamente.

//vertex buffer
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

//Index buffer
gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

Matrices escritas

WebGL proporciona un tipo especial de matriz llamado typed arrayspara transferir los elementos de datos como el vértice del índice y la textura. Estos arreglos escritos almacenan grandes cantidades de datos y los procesan en formato binario nativo, lo que da como resultado un mejor rendimiento. Las matrices escritas utilizadas por WebGL son Int8Array, Uint8Array, Int16Array, Uint16Array, Int32Array, UInt32Array, Float32Array y Float64Array.

Note

  • Generalmente, para almacenar datos de vértices, usamos Float32Array; y para almacenar datos de índice, utilizamosUint16Array.

  • Puede crear matrices escritas como matrices JavaScript usando new palabra clave.

Desvincular los búferes

Se recomienda desvincular los búferes después de usarlos. Se puede hacer pasando un valor nulo en lugar del objeto de búfer, como se muestra a continuación.

gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);

WebGL proporciona los siguientes métodos para realizar operaciones de búfer:

No Señor. Métodos y descripción
1

vacío bindBuffer( destino de enumeración , búfer de objeto )

objetivo : ARRAY_BUFFER, ELEMENT_ARRAY_BUFFER

2

vacío bufferData( destino de enumeración , tamaño largo , uso de enumeración )

objetivo : ARRAY_BUFFER, ELEMENT_ARRAY_BUFFER

uso - STATIC_DRAW, STREAM_DRAW, DYNAMIC_DRAW

3

vacío bufferData( destino de enumeración , datos de objeto , uso de enumeración )

destino y uso : igual que parabufferData encima

4

vacío bufferSubData( destino de enumeración , desplazamiento largo , datos de objeto )

objetivo : ARRAY_BUFFER, ELEMENT_ARRAY_BUFFER

5 Objeto createBuffer()
6 vacío deleteBuffer( Búfer de objeto )
7

ninguna getBufferParameter(enum target , enum pname )

objetivo - ARRAY_BUFFER, ELEMENT_ ARRAY_BUFFER

pname - BUFFER_SIZE, BUFFER_USAGE

8 bool isBuffer( Búfer de objeto )

Los sombreadores son los programas que se ejecutan en GPU. Los sombreadores están escritos en OpenGL ES Shader Language (conocido como ES SL). ES SL tiene variables propias, tipos de datos, calificadores, entradas y salidas integradas.

Tipos de datos

La siguiente tabla enumera los tipos de datos básicos proporcionados por OpenGL ES SL.

No Señor. Tipo y descripción
1

void

Representa un valor vacío.

2

bool

Acepta verdadero o falso.

3

int

Este es un tipo de datos entero con signo.

4

float

Este es un tipo de datos escalares flotantes.

5

vec2, vec3, vec4

vector de punto flotante de n componentes

6

bvec2, bvec3, bvec4

Vector booleano

7

ivec2, ivec3, ivec4

vector entero con signo

8

mat2, mat3, mat4

Matriz flotante 2x2, 3x3, 4x4

9

sampler2D

Accede a una textura 2D

10

samplerCube

Acceder a la textura mapeada del cubo

Calificadores

Hay tres clasificatorios principales en OpenGL ES SL:

No Señor. Calificador y descripción
1

attribute

Este calificador actúa como un enlace entre un sombreador de vértices y OpenGL ES para datos por vértice. El valor de este atributo cambia para cada ejecución del sombreador de vértices.

2

uniform

Este calificador vincula los programas de sombreado y la aplicación WebGL. A diferencia del calificador de atributo, los valores de los uniformes no cambian. Los uniformes son de solo lectura; puede usarlos con cualquier tipo de datos básico, para declarar una variable.

Example - uniforme vec4 lightPosition;

3

varying

Este calificador forma un vínculo entre un sombreador de vértices y un sombreador de fragmentos para datos interpolados. Se puede utilizar con los siguientes tipos de datos: float, vec2, vec3, vec4, mat2, mat3, mat4 o matrices.

Example - variando vec3 normal;

Sombreador de vértices

Vertex shader es un código de programa, que se llama en cada vértice. Transforma (mueve) la geometría (ej .: triángulo) de un lugar a otro. Maneja los datos de cada vértice (datos por vértice) como coordenadas de vértice, normales, colores y coordenadas de textura.

En el código ES GL del sombreador de vértices, los programadores tienen que definir atributos para manejar datos. Estos atributos apuntan a un objeto Vertex Buffer escrito en JavaScript. Las siguientes tareas se pueden realizar utilizando sombreadores de vértices junto con la transformación de vértices:

  • Transformación de vértice
  • Normalización y transformación normal
  • Generación de coordenadas de textura
  • Transformación de coordenadas de textura
  • Lighting
  • Aplicación de material de color

Variables predefinidas

OpenGL ES SL proporciona las siguientes variables predefinidas para el sombreador de vértices:

No Señor. Variables y descripción
1

highp vec4 gl_Position;

Mantiene la posición del vértice.

2

mediump float gl_PointSize;

Mantiene el tamaño de punto transformado. Las unidades de esta variable son píxeles.

Código de muestra

Eche un vistazo al siguiente código de muestra de un sombreador de vértices. Procesa los vértices de un triángulo.

attribute vec2 coordinates;

void main(void) {
   gl_Position = vec4(coordinates, 0.0, 1.0);
};

Si observa atentamente el código anterior, hemos declarado una variable de atributo con el nombre coordinates. (Esta variable se asociará con el objeto Vertex Buffer utilizando el métodogetAttribLocation(). El atributocoordinates se pasa como parámetro a este método junto con el objeto del programa de sombreado.)

En el segundo paso del programa de sombreado de vértices dado, el gl_position la variable está definida.

gl_Position

gl_Position es la variable predefinida que solo está disponible en el programa de sombreado de vértices. Contiene la posición del vértice. En el código anterior, elcoordinatesEl atributo se pasa en forma de vector. Como el sombreador de vértices es una operación por vértice, el valor gl_position se calcula para cada vértice.

Posteriormente, el valor gl_position es utilizado por el ensamblaje de primitivas, recorte, selección y otras operaciones de funcionalidad fija que operan en las primitivas una vez finalizado el procesamiento de vértices.

Podemos escribir programas de sombreado de vértices para todas las operaciones posibles del sombreador de vértices, que discutiremos individualmente en este tutorial.

Shader de fragmentos

UN mesh está formado por múltiples triángulos, y la superficie de cada triángulo se conoce como fragment. Un sombreador de fragmentos es el código que se ejecuta en cada píxel de cada fragmento. Esto está escrito para calcular y rellenar el color en píxeles individuales. Las siguientes tareas se pueden realizar utilizando sombreadores de fragmentos:

  • Operaciones sobre valores interpolados
  • Acceso a texturas
  • Aplicación de textura
  • Fog
  • Suma de colores

Variables predefinidas

OpenGL ES SL proporciona las siguientes variables predefinidas para el sombreador de fragmentos:

No Señor. Variables y descripción
1

mediump vec4 gl_FragCoord;

Mantiene la posición del fragmento dentro del búfer de fotogramas.

2

bool gl_FrontFacing;

Contiene el fragmento que pertenece a una primitiva frontal.

3

mediump vec2 gl_PointCoord;

Mantiene la posición del fragmento dentro de un punto (solo rasterización de puntos).

4

mediump vec4 gl_FragColor;

Mantiene el valor de color del fragmento de salida del sombreador

5

mediump vec4 gl_FragData[n]

Sostiene el color del fragmento para adjuntar el color n.

Código de muestra

El siguiente código de muestra de un sombreador de fragmentos muestra cómo aplicar color a cada píxel de un triángulo.

void main(void) {
   gl_FragColor = vec4(0, 0.8, 0, 1);
}

En el código anterior, el colorel valor se almacena en la variable gl.FragColor. El programa de sombreado de fragmentos pasa la salida a la canalización utilizando variables de función fija; FragColor es uno de ellos. Esta variable contiene el valor de color de los píxeles del modelo.

Almacenamiento y compilación de programas de sombreado

Dado que los sombreadores son programas independientes, podemos escribirlos como un script separado y usarlos en la aplicación. O puede almacenarlos directamente enstring formato, como se muestra a continuación.

var vertCode =
   'attribute vec2 coordinates;' +
	
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 0.0, 1.0);' +
   '}';

Compilando el Shader

La compilación implica seguir tres pasos:

  • Creando el objeto sombreador
  • Adjuntar el código fuente al objeto de sombreado creado
  • Compilando el programa

Creando el Vertex Shader

Para crear un sombreador vacío, WebGL proporciona un método llamado createShader(). Crea y devuelve el objeto de sombreado. Su sintaxis es la siguiente:

Object createShader (enum type)

Como se observa en la sintaxis, este método acepta un valor de enumeración predefinido como parámetro. Tenemos dos opciones para esto:

  • gl.VERTEX_SHADER para crear sombreadores de vértices

  • gl.FRAGMENT_SHADER para crear sombreadores de fragmentos.

Adjuntar la fuente al sombreador

Puede adjuntar el código fuente al objeto de sombreado creado utilizando el método shaderSource(). Su sintaxis es la siguiente:

void shaderSource(Object shader, string source)

Este método acepta dos parámetros:

  • shader - Tienes que pasar el objeto de sombreado creado como un parámetro.

  • Source - Tienes que pasar el código del programa de sombreado en formato de cadena.

Compilando el programa

Para compilar el programa, debes usar el método compileShader(). Su sintaxis es la siguiente:

compileShader(Object shader)

Este método acepta el objeto del programa de sombreado como parámetro. Después de crear un objeto de programa de sombreado, adjunte el código fuente y pase ese objeto a este método.

El siguiente fragmento de código muestra cómo crear y compilar un sombreador de vértices y un sombreador de fragmentos para crear un triángulo.

// Vertex Shader
var vertCode =
   'attribute vec3 coordinates;' +
	
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
   '}';

var vertShader = gl.createShader(gl.VERTEX_SHADER);
gl.shaderSource(vertShader, vertCode);
gl.compileShader(vertShader);
 
// Fragment Shader
var fragCode =
   'void main(void) {' +
      ' gl_FragColor = vec4(0, 0.8, 0, 1);' +
   '}';

var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
gl.shaderSource(fragShader, fragCode);
gl.compileShader(fragShader);

Programa combinado

Después de crear y compilar ambos programas de sombreado, debe crear un programa combinado que contenga ambos sombreadores (vértice y fragmento). Deben seguirse los siguientes pasos:

  • Crea un objeto de programa
  • Adjunte ambos sombreadores
  • Vincular ambos sombreadores
  • Utilizar el programa

Crear un objeto de programa

Crea un objeto de programa usando el método createProgram(). Devolverá un objeto de programa vacío. Aquí está su sintaxis:

createProgram();

Adjuntar los sombreadores

Adjunte los sombreadores al objeto de programa creado usando el método attachShader(). Su sintaxis es la siguiente:

attachShader(Object program, Object shader);

Este método acepta dos parámetros:

  • Program - Pase el objeto de programa vacío creado como un parámetro.

  • Shader - Pase uno de los programas de sombreadores compilados (sombreador de vértices, sombreador de fragmentos)

Note - Necesita adjuntar ambos sombreadores usando este método.

Vincular los sombreadores

Vincula los sombreadores usando el método linkProgram(), pasando el objeto de programa al que ha adjuntado los sombreadores. Su sintaxis es la siguiente:

linkProgram(shaderProgram);

Utilice el programa

WebGL proporciona un método llamado useProgram(). Debe pasarle el programa vinculado. Su sintaxis es la siguiente:

useProgram(shaderProgram);

El siguiente fragmento de código muestra cómo crear, vincular y utilizar un programa de sombreado combinado.

var shaderProgram = gl.createProgram();
gl.attachShader(shaderProgram, vertShader);
gl.attachShader(shaderProgram, fragShader);
gl.linkProgram(shaderProgram);
gl.useProgram(shaderProgram);

Cada atributo del programa de sombreado de vértices apunta a un objeto de búfer de vértice. Después de crear los objetos de búfer de vértices, los programadores deben asociarlos con los atributos del programa de sombreado de vértices. Cada atributo apunta a un solo objeto de búfer de vértice del que extraen los valores de datos, y luego estos atributos se pasan al programa de sombreado.

Para asociar los objetos de búfer de vértices con los atributos del programa de sombreado de vértices, debe seguir los pasos que se indican a continuación:

  • Obtener la ubicación del atributo
  • Apunta el atributo a un objeto de búfer de vértice
  • Habilita el atributo

Obtener la ubicación del atributo

WebGL proporciona un método llamado getAttribLocation()que devuelve la ubicación del atributo. Su sintaxis es la siguiente:

ulong getAttribLocation(Object program, string name)

Este método acepta el objeto de programa de sombreado de vértices y los valores de atributo del programa de sombreado de vértices.

El siguiente fragmento de código muestra cómo utilizar este método.

var coordinatesVar = gl.getAttribLocation(shader_program, "coordinates");

Aquí, shader_program es el objeto del programa de sombreado y coordinates es el atributo del programa de sombreado de vértices.

Apunta el atributo a un VBO

Para asignar el objeto de búfer a la variable de atributo, WebGL proporciona un método llamado vertexAttribPointer(). Aquí está la sintaxis de este método:

void vertexAttribPointer(location, int size, enum type, bool normalized, long stride, long offset)

Este método acepta seis parámetros y se analizan a continuación.

  • Location- Especifica la ubicación de almacenamiento de una variable de atributo. En esta opción, debe pasar el valor devuelto por elgetAttribLocation() método.

  • Size - Especifica el número de componentes por vértice en el objeto de búfer.

  • Type - Especifica el tipo de datos.

  • Normalized- Este es un valor booleano. Si es verdadero, los datos no flotantes se normalizan a [0, 1]; de lo contrario, se normaliza a [-1, 1].

  • Stride - Especifica el número de bytes entre diferentes elementos de datos de vértice, o cero para el paso predeterminado.

  • Offset- Especifica el desplazamiento (en bytes) en un objeto de búfer para indicar desde qué byte se almacenan los datos del vértice. Si los datos se almacenan desde el principio, el desplazamiento es 0.

El siguiente fragmento muestra cómo usar vertexAttribPointer() en un programa -

gl.vertexAttribPointer(coordinatesVar, 3, gl.FLOAT, false, 0, 0);

Habilitar el atributo

Active el atributo de sombreador de vértices para acceder al objeto de búfer en un sombreador de vértices. Para esta operación, WebGL proporcionaenableVertexAttribArray()método. Este método acepta la ubicación del atributo como parámetro. A continuación se explica cómo utilizar este método en un programa:

gl.enableVertexAttribArray(coordinatesVar);

Después de asociar los búferes con los sombreadores, el paso final es dibujar las primitivas requeridas. WebGL proporciona dos métodos, a saber,drawArrays() y drawElements() dibujar modelos.

drawArrays ()

drawArrays()es el método que se utiliza para dibujar modelos utilizando vértices. Aquí está su sintaxis:

void drawArrays(enum mode, int first, long count)

Este método toma los siguientes tres parámetros:

  • mode- En WebGL, los modelos se dibujan utilizando tipos primitivos. Al usar el modo, los programadores deben elegir uno de los tipos primitivos proporcionados por WebGL. Los valores posibles para esta opción son: gl.POINTS, gl.LINE_STRIP, gl.LINE_LOOP, gl.LINES, gl.TRIANGLE_STRIP, gl.TRIANGLE_FAN y gl.TRIANGLES.

  • first- Esta opción especifica el elemento inicial en las matrices habilitadas. No puede ser un valor negativo.

  • count - Esta opción especifica el número de elementos a renderizar.

Si dibuja un modelo usando drawArrays() , luego WebGL, mientras renderiza las formas, crea la geometría en el orden en que se definen las coordenadas del vértice.

Ejemplo

Si quieres dibujar un solo triángulo usando drawArray() método, entonces tienes que pasar tres vértices y llamar al drawArrays() método, como se muestra a continuación.

var vertices = [-0.5,-0.5, -0.25,0.5, 0.0,-0.5,];
gl.drawArrays(gl.TRIANGLES, 0, 3);

Producirá un triángulo como se muestra a continuación.

Suponga que desea dibujar triángulos contiguos, luego debe pasar los siguientes tres vértices en orden en el búfer de vértices y mencionar el número de elementos que se renderizarán como 6.

var vertices = [-0.5,-0.5, -0.25,0.5, 0.0,-0.5, 0.0,-0.5, 0.25,0.5, 0.5,-0.5,];
gl.drawArrays(gl.TRIANGLES, 0, 6);

Producirá un triángulo contiguo como se muestra a continuación.

drawElements ()

drawElements()es el método que se utiliza para dibujar modelos utilizando vértices e índices. Su sintaxis es la siguiente:

void drawElements(enum mode, long count, enum type, long offset)

Este método toma los siguientes cuatro parámetros:

  • mode- Los modelos WebGL se dibujan utilizando tipos primitivos. Al usar el modo, los programadores deben elegir uno de los tipos primitivos proporcionados por WebGL. La lista de valores posibles para esta opción son: gl.POINTS, gl.LINE_STRIP, gl.LINE_LOOP, gl.LINES, gl.TRIANGLE_STRIP, gl.TRIANGLE_FAN y gl.TRIANGLES.

  • count - Esta opción especifica el número de elementos a renderizar.

  • type - Esta opción especifica el tipo de datos de los índices que deben ser UNSIGNED_BYTE o UNSIGNED_SHORT.

  • offset- Esta opción especifica el punto de partida para la renderización. Suele ser el primer elemento (0).

Si dibuja un modelo usando drawElements(), entonces el objeto de búfer de índice también debe crearse junto con el objeto de búfer de vértice. Si usa este método, los datos de vértice se procesarán una vez y se usarán tantas veces como se mencione en los índices.

Ejemplo

Si desea dibujar un solo triángulo usando índices, debe pasar los índices junto con los vértices y llamar al drawElements() método como se muestra a continuación.

var vertices = [ -0.5,-0.5,0.0, -0.25,0.5,0.0, 0.0,-0.5,0.0 ];
var indices = [0,1,2];

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

Producirá la siguiente salida:

Si quieres dibujar triángulos contagiosos usando drawElements() método, simplemente agregue los otros vértices y mencione los índices de los vértices restantes.

var vertices = [
   -0.5,-0.5,0.0,
   -0.25,0.5,0.0,
   0.0,-0.5,0.0,
   0.25,0.5,0.0,
   0.5,-0.5,0.0 
];

var indices = [0,1,2,2,3,4];

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

Producirá la siguiente salida:

Operaciones requeridas

Antes de dibujar una primitiva, debe realizar algunas operaciones, que se explican a continuación.

Limpiar el lienzo

Primero que nada, debes limpiar el lienzo, usando clearColor()método. Puede pasar los valores RGBA de un color deseado como parámetro de este método. Luego, WebGL borra el lienzo y lo llena con el color especificado. Por lo tanto, puede utilizar este método para configurar el color de fondo.

Eche un vistazo al siguiente ejemplo. Aquí estamos pasando el valor RGBA del color gris.

gl.clearColor(0.5, 0.5, .5, 1);

Habilitar prueba de profundidad

Habilite la prueba de profundidad usando el enable() método, como se muestra a continuación.

gl.enable(gl.DEPTH_TEST);

Borrar el bit de búfer de color

Borre el color y el búfer de profundidad utilizando el clear() método, como se muestra a continuación.

gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

Configurar el puerto de vista

El puerto de visualización representa un área visible rectangular que contiene los resultados de la representación del búfer de dibujo. Puede establecer las dimensiones del puerto de visualización utilizandoviewport()método. En el siguiente código, las dimensiones del puerto de visualización se establecen en las dimensiones del lienzo.

gl.viewport(0,0,canvas.width,canvas.height);

Discutimos anteriormente (en el Capítulo 5) cómo seguir un proceso paso a paso para dibujar una primitiva. Hemos explicado el proceso en cinco pasos. Debe repetir estos pasos cada vez que dibuje una nueva forma. Este capítulo explica cómo dibujar puntos con coordenadas 3D en WebGL. Antes de continuar, echemos un vistazo a los cinco pasos.

Pasos requeridos

Los siguientes pasos son necesarios para crear una aplicación WebGL para dibujar puntos.

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

En este paso, obtenemos el objeto de contexto Renderizado WebGL usando el método getContext().

Step 2 − Define the Geometry and Store it in the Buffer Objects

Como estamos dibujando tres puntos, definimos tres vértices con coordenadas 3D y los almacenamos en zonas de influencia.

var vertices = [
   -0.5,0.5,0.0,
   0.0,0.5,0.0,
   -0.25,0.25,0.0, 
];

Step 3 − Create and Compile the Shader Programs

En este paso, debe escribir programas de sombreado de vértices y de fragmentos, compilarlos y crear un programa combinado vinculando estos dos programas.

  • Vertex Shader - En el sombreador de vértices del ejemplo dado, definimos un atributo de vector para almacenar coordenadas 3D, y lo asignamos al gl_position variable.

  • gl_pointsizees la variable que se utiliza para asignar un tamaño al punto. Asignamos el tamaño en puntos como 10.

var vertCode = 'attribute vec3 coordinates;' +

   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
      'gl_PointSize = 10.0;'+
   '}';
  • Fragment Shader - En el sombreador de fragmentos, simplemente asignamos el color del fragmento al gl_FragColor variable

var fragCode = 'void main(void) {' +' gl_FragColor = vec4(1, 0.5, 0.0, 1);' +'}';

Step 4 − Associate the Shader Programs to Buffer Objects

En este paso, asociamos los objetos de búfer con el programa de sombreado.

Step 5 − Drawing the Required Object

Usamos el método drawArrays()para dibujar puntos. Dado que la cantidad de puntos que queremos sacar es tres, el valor de conteo es 3.

gl.drawArrays(gl.POINTS, 0, 3)

Ejemplo: dibujar tres puntos con WebGL

Aquí está el programa WebGL completo para dibujar tres puntos:

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*================Creating a canvas=================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl'); 

         /*==========Defining and storing the geometry=======*/

         var vertices = [
            -0.5,0.5,0.0,
            0.0,0.5,0.0,
            -0.25,0.25,0.0, 
         ];

         // Create an empty buffer object to store the vertex buffer
         var vertex_buffer = gl.createBuffer();

         //Bind appropriate array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Pass the vertex data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         /*=========================Shaders========================*/

         // vertex shader source code
         var vertCode =
            'attribute vec3 coordinates;' +

            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
               'gl_PointSize = 10.0;'+
            '}';

         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         
         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);

         // fragment shader source code
         var fragCode =
            'void main(void) {' +
               ' gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragmentt shader
         gl.compileShader(fragShader);
         
         // Create a shader program object to store
         // the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader); 

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /*======== Associating shaders to buffer objects ========*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // Point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0);

         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         /*============= Drawing the primitive ===============*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);
 
         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawArrays(gl.POINTS, 0, 3);
      </script>
   </body>
</html>

Producirá el siguiente resultado:

En el capítulo anterior (Capítulo 11), discutimos cómo dibujar tres puntos usando WebGL. En el Capítulo 5, tomamos una aplicación de muestra para demostrar cómo dibujar un triángulo. En ambos ejemplos, hemos dibujado las primitivas usando solo vértices.

Para dibujar formas / mallas más complejas, también pasamos los índices de una geometría, junto con los vértices, a los sombreadores. En este capítulo, veremos cómo dibujar un triángulo usando índices.

Pasos necesarios para dibujar un triángulo

Se requieren los siguientes pasos para crear una aplicación WebGL para dibujar un triángulo.

Step 1 − Prepare the Canvas and Get WebGL Rendering Context

En este paso, obtenemos el objeto de contexto Renderizado WebGL usando getContext().

Step 2 − Define the Geometry and Store it in Buffer Objects

Dado que estamos dibujando un triángulo usando índices, tenemos que pasar los tres vértices del triángulo, incluidos los índices, y almacenarlos en los búferes.

var vertices = [
   -0.5,0.5,0.0,
   -0.5,-0.5,0.0,
   0.5,-0.5,0.0, 
];
	
indices = [0,1,2];

Step 3 − Create and Compile the Shader Programs

En este paso, debe escribir programas de sombreado de vértices y de fragmentos, compilarlos y crear un programa combinado vinculando estos dos programas.

  • Vertex Shader - En el sombreador de vértices del programa, definimos el atributo de vector para almacenar coordenadas 3D y asignarlo a gl_position.

var vertCode =
   'attribute vec3 coordinates;' +
	
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
   '}';
  • Fragment Shader - En el sombreador de fragmentos, simplemente asignamos el color del fragmento al gl_FragColor variable.

var fragCode = 'void main(void) {' +
   ' gl_FragColor = vec4(1, 0.5, 0.0, 1);' +
'}';

Step 4 − Associate the Shader Programs to the Buffer Objects

En este paso, asociamos los objetos de búfer y el programa de sombreado.

Step 5 − Drawing the Required Object

Dado que estamos dibujando un triángulo usando índices, usaremos drawElements(). A este método, tenemos que pasarle el número de índices. El valor de laindices.length significa el número de índices.

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

Ejemplo: dibujar un triángulo

El siguiente código de programa muestra cómo dibujar un triángulo en WebGL usando índices:

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*============== Creating a canvas ====================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');
      
         /*======== Defining and storing the geometry ===========*/

         var vertices = [
            -0.5,0.5,0.0,
            -0.5,-0.5,0.0,
            0.5,-0.5,0.0, 
         ];
         
         indices = [0,1,2];
         
         // Create an empty buffer object to store vertex buffer
         var vertex_buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         
         // Pass the vertex data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         // Create an empty buffer object to store Index buffer
         var Index_Buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);

         // Pass the vertex data to the buffer
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);
         
         // Unbind the buffer
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);

         /*================ Shaders ====================*/
         
         // Vertex shader source code
         var vertCode =
            'attribute vec3 coordinates;' +
				
            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
            '}';
            
         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);

         //fragment shader source code
         var fragCode =
            'void main(void) {' +
               ' gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';
            
         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode); 
         
         // Compile the fragmentt shader
         gl.compileShader(fragShader);

         // Create a shader program object to store
         // the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader);

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both the programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /*======= Associating shaders to buffer objects =======*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Bind index buffer object
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);
         
         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // Point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0); 
         
         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         /*=========Drawing the triangle===========*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);

         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);
      </script>
    </body>
</html>

Si ejecuta este ejemplo, producirá el siguiente resultado:

En el capítulo anterior (Capítulo 12), discutimos cómo dibujar un triángulo usando WebGL. Además de los triángulos, WebGL admite varios otros modos de dibujo. Este capítulo explica los modos de dibujo compatibles con WebGL.

El parámetro de modo

Echemos un vistazo a la sintaxis de los métodos: drawElements() y dibuja Arrays().

void drawElements(enum mode, long count, enum type, long offset);

void drawArrays(enum mode, int first, long count);

Si observa claramente, ambos métodos aceptan un parámetro mode. Con este parámetro, los programadores pueden seleccionar el modo de dibujo en WebGL.

Los modos de dibujo proporcionados por WebGL se enumeran en la siguiente tabla.

No Señor. Modo y descripción
1

gl.POINTS

Dibujar una serie de puntos.

2

gl.LINES

Para dibujar una serie de segmentos de línea no conectados (líneas individuales).

3

gl.LINE_STRIP

Para dibujar una serie de segmentos de línea conectados.

4

gl.LINE_LOOP

Para dibujar una serie de segmentos de línea conectados. También une el primer y último vértice para formar un bucle.

5

gl.TRIANGLES

Dibujar una serie de triángulos separados.

6

gl.TRIANGLE_STRIP

Dibujar una serie de triángulos conectados en forma de franjas.

7

gl.TRIANGLE_FAN

Dibujar una serie de triángulos conectados que compartan el primer vértice en forma de abanico.

Ejemplo: dibujar tres líneas paralelas

El siguiente ejemplo muestra cómo dibujar tres líneas paralelas usando gl.LINES.

<!doctype html>
<html>
   <body>
      <canvas width = "300" height = "300" id = "my_Canvas"></canvas>

      <script>
         /*======= Creating a canvas =========*/

         var canvas = document.getElementById('my_Canvas');
         var gl = canvas.getContext('experimental-webgl');

         /*======= Defining and storing the geometry ======*/

         var vertices = [
            -0.7,-0.1,0,
            -0.3,0.6,0,
            -0.3,-0.3,0,
            0.2,0.6,0,
            0.3,-0.3,0,
            0.7,0.6,0 
         ]

         // Create an empty buffer object
         var vertex_buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
      
         // Pass the vertex data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         /*=================== Shaders ====================*/

         // Vertex shader source code
         var vertCode =
            'attribute vec3 coordinates;' +
            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
            '}';

         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);

         // Fragment shader source code
         var fragCode =
            'void main(void) {' +
               'gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragmentt shader
         gl.compileShader(fragShader);

         // Create a shader program object to store
         // the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader);

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both the programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /*======= Associating shaders to buffer objects ======*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // Point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0);

         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         /*============ Drawing the triangle =============*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);

         // Clear the color and depth buffer
         gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawArrays(gl.LINES, 0, 6);

         // POINTS, LINE_STRIP, LINE_LOOP, LINES,
         // TRIANGLE_STRIP,TRIANGLE_FAN, TRIANGLES
      </script>
   </body>
</html>

Si ejecuta este ejemplo, producirá el siguiente resultado:

Modos de dibujo

En el programa anterior, si reemplaza el modo de drawArrays() con uno de los siguientes modos de dibujo, producirá diferentes salidas cada vez.

Modos de dibujo Salidas
LINE_STRIP
LINE_LOOP
TRIANGLE_STRIP
TRIANGLE_FAN
TRIANGULOS

En el capítulo anterior, discutimos los diferentes modos de dibujo proporcionados por WebGL. También podemos usar índices para dibujar primitivas usando uno de estos modos. Para dibujar modelos en WebGL, tenemos que elegir una de estas primitivas y dibujar la malla requerida (es decir, un modelo formado usando una o más primitivas).

En este capítulo, tomaremos un ejemplo para demostrar cómo dibujar un cuadrilátero usando WebGL.

Pasos para dibujar un cuadrilátero

Se requieren los siguientes pasos para crear una aplicación WebGL para dibujar un cuadrilátero.

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

En este paso, obtenemos el objeto de contexto Renderizado WebGL usando getContext().

Step 2 − Define the Geometry and Store it in the Buffer Objects

Se puede dibujar un cuadrado usando dos triángulos. En este ejemplo, proporcionamos los vértices de dos triángulos (con un borde común) e índices.

var vertices = [
   -0.5,0.5,0.0,
   -0.5,-0.5,0.0,
   0.5,-0.5,0.0,
   0.5,0.5,0.0 
];

indices = [3,2,1,3,1,0];

Step 3 − Create and Compile the Shader Programs

En este paso, debe escribir los programas de sombreado de vértices y de fragmentos, compilarlos y crear un programa combinado vinculando estos dos programas.

  • Vertex Shader - En el sombreador de vértices del programa, definimos el atributo de vector para almacenar coordenadas 3D y asignarlo a gl_position.

var vertCode =
   'attribute vec3 coordinates;' +
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
   '}';
  • Fragment Shader - En el sombreador de fragmentos, simplemente asignamos el color del fragmento al gl_FragColor variable.

var fragCode = 'void main(void) {' +' gl_FragColor = vec4(0.5, 0.3, 0.0, 7.5);' +'}';

Step 4 − Associate the Shader Programs to Buffer Objects

En este paso, asociamos los objetos de búfer con el programa de sombreado.

Step 5 − Drawing the Required Object

Dado que estamos dibujando dos triángulos para formar un cuádruple, usando índices, usaremos el método drawElements(). A este método, tenemos que pasarle el número de índices. El valor deindices.length da el número de índices.

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

Ejemplo: dibujar un cuadrilátero

El siguiente programa muestra cómo crear una aplicación WebGL para dibujar un cuadrilátero.

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*============ Creating a canvas =================*/
      
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');
      
         /*========== Defining and storing the geometry =========*/

         var vertices = [
            -0.5,0.5,0.0,
            -0.5,-0.5,0.0,
            0.5,-0.5,0.0,
            0.5,0.5,0.0 
         ];

         indices = [3,2,1,3,1,0];

         // Create an empty buffer object to store vertex buffer
         var vertex_buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Pass the vertex data to the buffer
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         // Create an empty buffer object to store Index buffer
         var Index_Buffer = gl.createBuffer();

         // Bind appropriate array buffer to it
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);

         // Pass the vertex data to the buffer
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

         // Unbind the buffer
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);

         /*====================== Shaders =======================*/

         // Vertex shader source code
         var vertCode =
            'attribute vec3 coordinates;' +
            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
            '}';

         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);

         // Fragment shader source code
         var fragCode =
            'void main(void) {' +
               ' gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         // Create fragment shader object 
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragmentt shader
         gl.compileShader(fragShader);

         // Create a shader program object to
         // store the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader);

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both the programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /* ======= Associating shaders to buffer objects =======*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Bind index buffer object
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer); 

         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // Point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0);

         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         /*============= Drawing the Quad ================*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);

         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         // Draw the triangle
         gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);
      </script>
   </body>
</html>

Si ejecuta este ejemplo, producirá el siguiente resultado:

En todos nuestros ejemplos anteriores, aplicamos color al objeto asignando un valor de color deseado al gl_FragColorvariable. Además de eso, podemos definir colores para cada vértice, al igual que las coordenadas y los índices de los vértices. Este capítulo toma un ejemplo para demostrar cómo aplicar colores a un cuadrilátero usando WebGL.

Aplicar colores

Para aplicar colores, debe definir los colores para cada vértice usando los valores RGB, en una matriz de JavaScript. Puede asignar los mismos valores a todos los vértices para tener un color único para el objeto. Después de definir los colores, debe crear un búfer de color y almacenar estos valores en él, y asociarlo a los atributos del sombreador de vértices.

En el sombreador de vértices, junto con el atributo de coordenadas (que mantiene la posición de los vértices), definimos un attribute y un varying manejar colores.

los color atributo contiene el valor de color por vértice, y varyinges la variable que se pasa como entrada al sombreador de fragmentos. Por tanto, tenemos que asignar elcolor valor para varying.

En el sombreador de fragmentos, el varying que contiene el valor de color se asigna a gl_FragColor, que tiene el color final del objeto.

Pasos para aplicar colores

Los siguientes pasos son necesarios para crear una aplicación WebGL para dibujar un Quad y aplicarle colores.

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

En este paso, obtenemos el objeto de contexto Renderizado WebGL usando getContext().

Step 2 − Define the Geometry and Store it in the Buffer Objects

Se puede dibujar un cuadrado usando dos triángulos. Por lo tanto, en este ejemplo, proporcionamos los vértices de dos triángulos (con un borde común) e índices. Como queremos aplicarle colores, también se define una variable que contiene los valores de color y se le asignan los valores de color para cada uno (rojo, azul, verde y rosa).

var vertices = [
   -0.5,0.5,0.0,
   -0.5,-0.5,0.0, 
   0.5,-0.5,0.0,
   0.5,0.5,0.0 
];

var colors = [ 0,0,1, 1,0,0, 0,1,0, 1,0,1,];
indices = [3,2,1,3,1,0];

Step 3 − Create and Compile the Shader Programs

En este paso, debe escribir los programas de sombreado de vértices y de fragmentos, compilarlos y crear un programa combinado vinculando estos dos programas.

  • Vertex Shader- En el sombreador de vértices del programa, definimos atributos vectoriales para almacenar coordenadas 3D (posición) y el color de cada vértice. UNvaringLa variable se declara para pasar los valores de color del sombreador de vértices al sombreador de fragmentos. Y finalmente, el valor almacenado en el atributo de color se asigna avarying.

var vertCode = 'attribute vec3 coordinates;'+
   'attribute vec3 color;'+
   'varying vec3 vColor;'+
	
   'void main(void) {' +
      ' gl_Position = vec4(coordinates, 1.0);' +
      'vColor = color;'+
   '}';
  • Fragment Shader - En el sombreador de fragmentos, asignamos el varying al gl_FragColor variable.

var fragCode = 'precision mediump float;'+
   'varying vec3 vColor;'+
   'void main(void) {'+
      'gl_FragColor = vec4(vColor, 1.);'+
   '}';

Step 4 − Associate the Shader Programs with the Buffer Objects

En este paso, asociamos los objetos de búfer y el programa de sombreado.

Step 5 − Drawing the Required Object

Como estamos dibujando dos triángulos que formarán un cuadrante, usando índices, usaremos el método drawElements(). A este método, tenemos que pasarle el número de índices. El valor deindices.length indica el número de índices.

gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);

Ejemplo: aplicación de color

El siguiente programa demuestra cómo dibujar un cuadrante usando la aplicación WebGL y aplicarle colores.

<!doctype html>
<html>
   <body>
    <canvas width = "300" height = "300" id = "my_Canvas"></canvas>

      <script>
         /*============= Creating a canvas ==================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');
         
         /*========== Defining and storing the geometry ==========*/

         var vertices = [
            -0.5,0.5,0.0,
            -0.5,-0.5,0.0,
            0.5,-0.5,0.0,
            0.5,0.5,0.0
         ];

         var colors = [0,0,1, 1,0,0, 0,1,0, 1,0,1,];
         
         indices = [3,2,1,3,1,0];
         
         // Create an empty buffer object and store vertex data
         var vertex_buffer = gl.createBuffer();
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);
         gl.bindBuffer(gl.ARRAY_BUFFER, null);

         // Create an empty buffer object and store Index data
         var Index_Buffer = gl.createBuffer();
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);

         // Create an empty buffer object and store color data
         var color_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);

         /*======================= Shaders =======================*/
         
         // vertex shader source code
         var vertCode = 'attribute vec3 coordinates;'+
            'attribute vec3 color;'+
            'varying vec3 vColor;'+
            'void main(void) {' +
               ' gl_Position = vec4(coordinates, 1.0);' +
               'vColor = color;'+
            '}';
            
         // Create a vertex shader object
         var vertShader = gl.createShader(gl.VERTEX_SHADER);

         // Attach vertex shader source code
         gl.shaderSource(vertShader, vertCode);

         // Compile the vertex shader
         gl.compileShader(vertShader);


         // fragment shader source code
         var fragCode = 'precision mediump float;'+
            'varying vec3 vColor;'+
            'void main(void) {'+
               'gl_FragColor = vec4(vColor, 1.);'+
            '}';
            
         // Create fragment shader object
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);

         // Attach fragment shader source code
         gl.shaderSource(fragShader, fragCode);

         // Compile the fragmentt shader
         gl.compileShader(fragShader);

         // Create a shader program object to
         // store the combined shader program
         var shaderProgram = gl.createProgram();

         // Attach a vertex shader
         gl.attachShader(shaderProgram, vertShader);

         // Attach a fragment shader
         gl.attachShader(shaderProgram, fragShader);

         // Link both the programs
         gl.linkProgram(shaderProgram);

         // Use the combined shader program object
         gl.useProgram(shaderProgram);

         /* ======== Associating shaders to buffer objects =======*/

         // Bind vertex buffer object
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         // Bind index buffer object
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, Index_Buffer);

         // Get the attribute location
         var coord = gl.getAttribLocation(shaderProgram, "coordinates");

         // point an attribute to the currently bound VBO
         gl.vertexAttribPointer(coord, 3, gl.FLOAT, false, 0, 0);

         // Enable the attribute
         gl.enableVertexAttribArray(coord);

         // bind the color buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         
         // get the attribute location
         var color = gl.getAttribLocation(shaderProgram, "color");
 
         // point attribute to the volor buffer object
         gl.vertexAttribPointer(color, 3, gl.FLOAT, false,0,0) ;
 
         // enable the color attribute
         gl.enableVertexAttribArray(color);

         /*============Drawing the Quad====================*/

         // Clear the canvas
         gl.clearColor(0.5, 0.5, 0.5, 0.9);

         // Enable the depth test
         gl.enable(gl.DEPTH_TEST);

         // Clear the color buffer bit
         gl.clear(gl.COLOR_BUFFER_BIT);

         // Set the view port
         gl.viewport(0,0,canvas.width,canvas.height);

         //Draw the triangle
         gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT,0);
      </script>
   </body>
</html>

Si ejecuta este ejemplo, producirá el siguiente resultado:

Hasta ahora, discutimos cómo dibujar varias formas y aplicar colores en ellas usando WebGL. Aquí, en este capítulo, tomaremos un ejemplo para mostrar cómo traducir un triángulo.

Traducción

La traducción es uno de los affine transformationsproporcionado por WebGL. Usando la traslación, podemos mover un triángulo (cualquier objeto) en el plano xyz. Supongamos que tenemos un triángulo [a, b, c] y queremos mover el triángulo a una posición que es 5 unidades hacia el eje X positivo y 3 unidades hacia el eje Y positivo. Entonces los nuevos vértices serían [a + 5, b + 3, c + 0]. Eso significa que, para trasladar el triángulo, necesitamos sumar las distancias de traslación, digamos, tx, ty, tz a cada vértice.

Dado que es un per-vertex operation, podemos llevarlo en el programa de sombreado de vértices.

En el sombreador de vértices, junto con el atributo, coordinates(que mantienen las posiciones de los vértices), definimos una variable uniforme que contiene las distancias de traslación (x, y, z). Posteriormente, agregamos esta variable uniforme a la variable de coordenadas y asignamos el resultado a lagl_Position variable.

Note - Dado que el sombreador de vértices se ejecutará en cada vértice, todos los vértices del triángulo se traducirán.

Pasos para trasladar un triángulo

Se requieren los siguientes pasos para crear una aplicación WebGL para dibujar un triángulo y luego trasladarlo a una nueva posición.

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

En este paso, obtenemos el objeto de contexto Renderizado WebGL usando getContext().

Step 2 − Define the Geometry and Store it in the Buffer Objects

Como estamos dibujando un triángulo, tenemos que pasar tres vértices del triángulo y almacenarlos en zonas de influencia.

var vertices = [ -0.5,0.5,0.0, -0.5,-0.5,0.0, 0.5,-0.5,0.0, ];

Step 3 − Create and Compile the Shader Programs

En este paso, debe escribir los programas de sombreado de vértices y de fragmentos, compilarlos y crear un programa combinado vinculando estos dos programas.

  • Vertex Shader- En el sombreador de vértices del programa, definimos un atributo de vector para almacenar coordenadas 3D. Junto con ella, definimos una variable uniforme para almacenar las distancias de traslación, y finalmente, sumamos estos dos valores y la asignamos agl_position que mantiene la posición final de los vértices.

var vertCode =
   'attribute vec4 coordinates;' +
   'uniform vec4 translation;'+
   'void main(void) {' +
      ' gl_Position = coordinates + translation;' +
   '}';
  • Fragment Shader - En el sombreador de fragmentos, simplemente asignamos el color del fragmento a la variable gl_FragColor.

var fragCode = 'void main(void) {' +' gl_FragColor = vec4(1, 0.5, 0.0, 1);' +'}';

Step 4 − Associate the Shader Programs to the Buffer Objects

En este paso, asociamos los objetos de búfer con el programa de sombreado.

Step 5 − Drawing the Required Object

Como estamos dibujando el triángulo usando índices, usaremos el método drawArrays(). A este método tenemos que pasarle el número de vértices / elementos a considerar. Como estamos dibujando un triángulo, pasaremos 3 como parámetro.

gl.drawArrays(gl.TRIANGLES, 0, 3);

Ejemplo: traducir un triángulo

El siguiente ejemplo muestra cómo trasladar un triángulo en el plano xyz.

<!doctype html>
<html>
   <body>
      <canvas width = "300" height = "300" id = "my_Canvas"></canvas>
         
      <script>
         /*=================Creating a canvas=========================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl'); 
 
         /*===========Defining and storing the geometry==============*/
         var vertices = [
            -0.5,0.5,0.0, 	
            -0.5,-0.5,0.0, 	
            0.5,-0.5,0.0,   
         ];
            
         //Create an empty buffer object and store vertex data            
         var vertex_buffer = gl.createBuffer(); 
			
         //Create a new buffer
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);   
			
         //bind it to the current buffer			
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW); 
			
         // Pass the buffer data
         gl.bindBuffer(gl.ARRAY_BUFFER, null);  
            
         /*========================Shaders============================*/
            
         //vertex shader source code 
         var vertCode =
            'attribute vec4 coordinates;' + 
            'uniform vec4 translation;'+
            'void main(void) {' +
               '  gl_Position = coordinates + translation;' +
            '}';
            
         //Create a vertex shader program object and compile it              
         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);
            
   
         //fragment shader source code
         var fragCode =
            'void main(void) {' +
               '   gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         //Create a fragment shader program object and compile it            
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);
            
         //Create and use combiened shader program
         var shaderProgram = gl.createProgram();
         gl.attachShader(shaderProgram, vertShader);
         gl.attachShader(shaderProgram, fragShader);
         gl.linkProgram(shaderProgram);
   
         gl.useProgram(shaderProgram);
   
         /* ===========Associating shaders to buffer objects============*/
      
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);    
         var coordinatesVar = gl.getAttribLocation(shaderProgram, "coordinates");
         gl.vertexAttribPointer(coordinatesVar, 3, gl.FLOAT, false, 0, 0);   
         gl.enableVertexAttribArray(coordinatesVar); 
   
         /* ==========translation======================================*/
         var Tx = 0.5, Ty = 0.5, Tz = 0.0;
         var translation = gl.getUniformLocation(shaderProgram, 'translation');
         gl.uniform4f(translation, Tx, Ty, Tz, 0.0);
 
         /*=================Drawing the riangle and transforming it========================*/ 

         gl.clearColor(0.5, 0.5, 0.5, 0.9);
         gl.enable(gl.DEPTH_TEST);
   
         gl.clear(gl.COLOR_BUFFER_BIT);
         gl.viewport(0,0,canvas.width,canvas.height);
         gl.drawArrays(gl.TRIANGLES, 0, 3);
      </script>
    </body>
 </html>

Si ejecuta este ejemplo, producirá el siguiente resultado:

En este capítulo, tomaremos un ejemplo para demostrar cómo modificar la escala de un triángulo usando WebGL.

Escalada

Escalar no es más que aumentar o disminuir el tamaño de un objeto. Por ejemplo, si un triángulo tiene vértices del tamaño [a, b, c], entonces el triángulo con los vértices [2a, 2b, 2c] tendrá el doble de tamaño. Por lo tanto, para escalar un triángulo, debes multiplicar cada vértice con el factor de escala. También puede escalar un vértice en particular.

Para escalar un triángulo, en el sombreador de vértices del programa, creamos una matriz uniforme y multiplicamos los valores de las coordenadas con esta matriz. Posteriormente, pasamos una matriz diagonal de 4 × 4 que tiene los factores de escala de las coordenadas x, y, z en las posiciones diagonales (última posición diagonal 1).

Pasos requeridos

Se requieren los siguientes pasos para crear una aplicación WebGL para escalar un triángulo.

Step 1 − Prepare the Canvas and Get the WebGL Rendering Context

En este paso, obtenemos el objeto de contexto Renderizado WebGL usando getContext().

Step 2 − Define the Geometry and Store it in the Buffer Objects

Como estamos dibujando un triángulo, tenemos que pasar tres vértices del triángulo y almacenarlos en zonas de influencia.

var vertices = [ -0.5,0.5,0.0, -0.5,-0.5,0.0, 0.5,-0.5,0.0, ];

Step 3 − Create and Compile the Shader Programs

En este paso, debe escribir los programas de sombreado de vértices y de fragmentos, compilarlos y crear un programa combinado vinculando estos dos programas.

  • Vertex Shader- En el sombreador de vértices del programa, definimos un atributo de vector para almacenar coordenadas 3D. Junto con él, definimos una matriz uniforme para almacenar los factores de escala, y finalmente, multiplicamos estos dos valores y lo asignamos agl_position que mantiene la posición final de los vértices.

var vertCode =
   'attribute vec4 coordinates;' +
   'uniform mat4 u_xformMatrix;' +
   'void main(void) {' +
      ' gl_Position = u_xformMatrix * coordinates;' +
   '}';
  • Fragment Shader - En el sombreador de fragmentos, simplemente asignamos el color del fragmento al gl_FragColor variable.

var fragCode = 'void main(void) {' +' gl_FragColor = vec4(1, 0.5, 0.0, 1);' +'}';

Step 4 − Associate the Shader Programs with the Buffer Objects

En este paso, asociamos los objetos de búfer con el programa de sombreado.

Step 5 − Drawing the Required Object

Como dibujamos el triángulo usando índices, usamos el drawArrays()método. A este método tenemos que pasarle el número de vértices / elementos a considerar. Como estamos dibujando un triángulo, pasaremos 3 como parámetro.

gl.drawArrays(gl.TRIANGLES, 0, 3);

Ejemplo: escalar un triángulo

El siguiente ejemplo muestra cómo escalar un triángulo:

<!doctype html>
<html>
   <body>
      <canvas width = "300" height = "300" id = "my_Canvas"></canvas>

      <script>
         /*=================Creating a canvas=========================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl'); 

         /*===========Defining and storing the geometry==============*/
         var vertices =  [
            -0.5,0.5,0.0, 	
            -0.5,-0.5,0.0, 	
            0.5,-0.5,0.0,   
         ];

         //Create an empty buffer object and store vertex data

         var vertex_buffer = gl.createBuffer();                                                     
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);                                                
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);           
         gl.bindBuffer(gl.ARRAY_BUFFER, null);  

         /*========================Shaders============================*/

         //Vertex shader source code
         var vertCode =
            'attribute vec4 coordinates;' + 
            'uniform mat4 u_xformMatrix;' +
            'void main(void) {' +
               '  gl_Position = u_xformMatrix * coordinates;' +
            '}';

         //Create a vertex shader program object and compile it                
         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);

         //fragment shader source code
         var fragCode =
            'void main(void) {' +
               '   gl_FragColor = vec4(0.0, 0.0, 0.0, 0.1);' +
            '}';

         //Create a fragment shader program object and compile it 
         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);

         //Create and use combiened shader program
         var shaderProgram = gl.createProgram();
         gl.attachShader(shaderProgram, vertShader);
         gl.attachShader(shaderProgram, fragShader);
         gl.linkProgram(shaderProgram);

         gl.useProgram(shaderProgram); 

         /*===================scaling==========================*/

         var Sx = 1.0, Sy = 1.5, Sz = 1.0;
         var xformMatrix = new Float32Array([
            Sx,   0.0,  0.0,  0.0,
            0.0,  Sy,   0.0,  0.0,
            0.0,  0.0,  Sz,   0.0,
            0.0,  0.0,  0.0,  1.0  
         ]);

         var u_xformMatrix = gl.getUniformLocation(shaderProgram, 'u_xformMatrix');
         gl.uniformMatrix4fv(u_xformMatrix, false, xformMatrix);

         /* ===========Associating shaders to buffer objects============*/
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);   

         var coordinatesVar = gl.getAttribLocation(shaderProgram, "coordinates"); 
         gl.vertexAttribPointer(coordinatesVar, 3, gl.FLOAT, false, 0, 0);  
         gl.enableVertexAttribArray(coordinatesVar);

         /*=================Drawing the Quad========================*/ 
         gl.clearColor(0.5, 0.5, 0.5, 0.9);
         gl.enable(gl.DEPTH_TEST);

         gl.clear(gl.COLOR_BUFFER_BIT);
         gl.viewport(0,0,canvas.width,canvas.height);
         gl.drawArrays(gl.TRIANGLES, 0, 3);
      </script>
   </body>
</html>

Si ejecuta este ejemplo, producirá el siguiente resultado:

En este capítulo, tomaremos un ejemplo para demostrar cómo rotar un triángulo usando WebGL.

Ejemplo: rotar un triángulo

El siguiente programa muestra cómo rotar un triángulo usando WebGL.

<!doctype html>
<html>
   <body>
      <canvas width = "400" height = "400" id = "my_Canvas"></canvas>

      <script>
         /*=================Creating a canvas=========================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');

         /*===========Defining and storing the geometry==============*/

         var vertices = [ -1,-1,-1, 1,-1,-1, 1, 1,-1 ];
         var colors = [ 1,1,1, 1,1,1, 1,1,1 ];
         var indices = [ 0,1,2 ];

         //Create and store data into vertex buffer
         var vertex_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         //Create and store data into color buffer
         var color_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);

         //Create and store data into index buffer
         var index_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

         /*==========================Shaders=========================*/

         var vertCode = 'attribute vec3 position;'+
            'uniform mat4 Pmatrix;'+
            'uniform mat4 Vmatrix;'+
            'uniform mat4 Mmatrix;'+
            'attribute vec3 color;'+//the color of the point
            'varying vec3 vColor;'+

            'void main(void) { '+//pre-built function
               'gl_Position = Pmatrix*Vmatrix*Mmatrix*vec4(position, 1.);'+
               'vColor = color;'+
            '}';

         var fragCode = 'precision mediump float;'+
            'varying vec3 vColor;'+
            'void main(void) {'+
               'gl_FragColor = vec4(vColor, 1.);'+
            '}';

         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);

         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);

         var shaderProgram = gl.createProgram();
         gl.attachShader(shaderProgram, vertShader);
         gl.attachShader(shaderProgram, fragShader);
         gl.linkProgram(shaderProgram);

         /*===========associating attributes to vertex shader ============*/

         var Pmatrix = gl.getUniformLocation(shaderProgram, "Pmatrix");
         var Vmatrix = gl.getUniformLocation(shaderProgram, "Vmatrix");
         var Mmatrix = gl.getUniformLocation(shaderProgram, "Mmatrix");
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);

         var position = gl.getAttribLocation(shaderProgram, "position");
         gl.vertexAttribPointer(position, 3, gl.FLOAT, false,0,0) ; //position
         gl.enableVertexAttribArray(position);
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);

         var color = gl.getAttribLocation(shaderProgram, "color");
         gl.vertexAttribPointer(color, 3, gl.FLOAT, false,0,0) ; //color
         gl.enableVertexAttribArray(color);
         gl.useProgram(shaderProgram);

         /*========================= MATRIX ========================= */

         function get_projection(angle, a, zMin, zMax) {
            var ang = Math.tan((angle*.5)*Math.PI/180);//angle*.5
            return [
               0.5/ang, 0 , 0, 0,
               0, 0.5*a/ang, 0, 0,
               0, 0, -(zMax+zMin)/(zMax-zMin), -1,
               0, 0, (-2*zMax*zMin)/(zMax-zMin), 0
            ];
         }

         var proj_matrix = get_projection(40, canvas.width/canvas.height, 1, 100);
         var mov_matrix = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];
         var view_matrix = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];

         //translating z
         view_matrix[14] = view_matrix[14]-6; //zoom

         /*=======================rotation========================*/
         function rotateZ(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv0 = m[0], mv4 = m[4], mv8 = m[8]; 

            m[0] = c*m[0]-s*m[1];
            m[4] = c*m[4]-s*m[5];
            m[8] = c*m[8]-s*m[9];
            m[1] = c*m[1]+s*mv0;
            m[5] = c*m[5]+s*mv4;
            m[9] = c*m[9]+s*mv8;
         }

         /*=================Drawing===========================*/

         var time_old = 0;
         var animate = function(time) {
            var dt = time-time_old;
            rotateZ(mov_matrix, dt*0.002);
            time_old = time;

            gl.enable(gl.DEPTH_TEST);
            gl.depthFunc(gl.LEQUAL);
            gl.clearColor(0.5, 0.5, 0.5, 0.9);
            gl.clearDepth(1.0);
            gl.viewport(0.0, 0.0, canvas.width, canvas.height);
            gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

            gl.uniformMatrix4fv(Pmatrix, false, proj_matrix);
            gl.uniformMatrix4fv(Vmatrix, false, view_matrix);
            gl.uniformMatrix4fv(Mmatrix, false, mov_matrix);

            gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
            gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT, 0);
            window.requestAnimationFrame(animate);
         }
         animate(0);
      </script>
   </body>
</html>

Si ejecuta este ejemplo, producirá el siguiente resultado:

En este capítulo, tomaremos un ejemplo para demostrar cómo dibujar un cubo 3D giratorio usando WebGL.

Ejemplo: dibujar un cubo 3D giratorio

El siguiente programa muestra cómo dibujar un cubo 3D giratorio:

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*============= Creating a canvas =================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');

         /*============ Defining and storing the geometry =========*/

         var vertices = [
            -1,-1,-1, 1,-1,-1, 1, 1,-1, -1, 1,-1,
            -1,-1, 1, 1,-1, 1, 1, 1, 1, -1, 1, 1,
            -1,-1,-1, -1, 1,-1, -1, 1, 1, -1,-1, 1,
            1,-1,-1, 1, 1,-1, 1, 1, 1, 1,-1, 1,
            -1,-1,-1, -1,-1, 1, 1,-1, 1, 1,-1,-1,
            -1, 1,-1, -1, 1, 1, 1, 1, 1, 1, 1,-1, 
         ];

         var colors = [
            5,3,7, 5,3,7, 5,3,7, 5,3,7,
            1,1,3, 1,1,3, 1,1,3, 1,1,3,
            0,0,1, 0,0,1, 0,0,1, 0,0,1,
            1,0,0, 1,0,0, 1,0,0, 1,0,0,
            1,1,0, 1,1,0, 1,1,0, 1,1,0,
            0,1,0, 0,1,0, 0,1,0, 0,1,0
         ];

         var indices = [
            0,1,2, 0,2,3, 4,5,6, 4,6,7,
            8,9,10, 8,10,11, 12,13,14, 12,14,15,
            16,17,18, 16,18,19, 20,21,22, 20,22,23 
         ];

         // Create and store data into vertex buffer
         var vertex_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Create and store data into color buffer
         var color_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);

         // Create and store data into index buffer
         var index_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

         /*=================== Shaders =========================*/

         var vertCode = 'attribute vec3 position;'+
            'uniform mat4 Pmatrix;'+
            'uniform mat4 Vmatrix;'+
            'uniform mat4 Mmatrix;'+
            'attribute vec3 color;'+//the color of the point
            'varying vec3 vColor;'+

            'void main(void) { '+//pre-built function
               'gl_Position = Pmatrix*Vmatrix*Mmatrix*vec4(position, 1.);'+
               'vColor = color;'+
            '}';

         var fragCode = 'precision mediump float;'+
            'varying vec3 vColor;'+
            'void main(void) {'+
               'gl_FragColor = vec4(vColor, 1.);'+
            '}';

         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);

         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);

         var shaderProgram = gl.createProgram();
         gl.attachShader(shaderProgram, vertShader);
         gl.attachShader(shaderProgram, fragShader);
         gl.linkProgram(shaderProgram);

         /* ====== Associating attributes to vertex shader =====*/
         var Pmatrix = gl.getUniformLocation(shaderProgram, "Pmatrix");
         var Vmatrix = gl.getUniformLocation(shaderProgram, "Vmatrix");
         var Mmatrix = gl.getUniformLocation(shaderProgram, "Mmatrix");

         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         var position = gl.getAttribLocation(shaderProgram, "position");
         gl.vertexAttribPointer(position, 3, gl.FLOAT, false,0,0) ;

         // Position
         gl.enableVertexAttribArray(position);
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         var color = gl.getAttribLocation(shaderProgram, "color");
         gl.vertexAttribPointer(color, 3, gl.FLOAT, false,0,0) ;

         // Color
         gl.enableVertexAttribArray(color);
         gl.useProgram(shaderProgram);

         /*==================== MATRIX =====================*/

         function get_projection(angle, a, zMin, zMax) {
            var ang = Math.tan((angle*.5)*Math.PI/180);//angle*.5
            return [
               0.5/ang, 0 , 0, 0,
               0, 0.5*a/ang, 0, 0,
               0, 0, -(zMax+zMin)/(zMax-zMin), -1,
               0, 0, (-2*zMax*zMin)/(zMax-zMin), 0 
            ];
         }

         var proj_matrix = get_projection(40, canvas.width/canvas.height, 1, 100);

         var mov_matrix = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];
         var view_matrix = [1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1];

         // translating z
         view_matrix[14] = view_matrix[14]-6;//zoom

         /*==================== Rotation ====================*/

         function rotateZ(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv0 = m[0], mv4 = m[4], mv8 = m[8];

            m[0] = c*m[0]-s*m[1];
            m[4] = c*m[4]-s*m[5];
            m[8] = c*m[8]-s*m[9];

            m[1]=c*m[1]+s*mv0;
            m[5]=c*m[5]+s*mv4;
            m[9]=c*m[9]+s*mv8;
         }

         function rotateX(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv1 = m[1], mv5 = m[5], mv9 = m[9];

            m[1] = m[1]*c-m[2]*s;
            m[5] = m[5]*c-m[6]*s;
            m[9] = m[9]*c-m[10]*s;

            m[2] = m[2]*c+mv1*s;
            m[6] = m[6]*c+mv5*s;
            m[10] = m[10]*c+mv9*s;
         }

         function rotateY(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv0 = m[0], mv4 = m[4], mv8 = m[8];

            m[0] = c*m[0]+s*m[2];
            m[4] = c*m[4]+s*m[6];
            m[8] = c*m[8]+s*m[10];

            m[2] = c*m[2]-s*mv0;
            m[6] = c*m[6]-s*mv4;
            m[10] = c*m[10]-s*mv8;
         }

         /*================= Drawing ===========================*/
         var time_old = 0;

         var animate = function(time) {

            var dt = time-time_old;
            rotateZ(mov_matrix, dt*0.005);//time
            rotateY(mov_matrix, dt*0.002);
            rotateX(mov_matrix, dt*0.003);
            time_old = time;

            gl.enable(gl.DEPTH_TEST);
            gl.depthFunc(gl.LEQUAL);
            gl.clearColor(0.5, 0.5, 0.5, 0.9);
            gl.clearDepth(1.0);

            gl.viewport(0.0, 0.0, canvas.width, canvas.height);
            gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);
            gl.uniformMatrix4fv(Pmatrix, false, proj_matrix);
            gl.uniformMatrix4fv(Vmatrix, false, view_matrix);
            gl.uniformMatrix4fv(Mmatrix, false, mov_matrix);
            gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
            gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT, 0);

            window.requestAnimationFrame(animate);
         }
         animate(0);
      </script>
   </body>
</html>

Si ejecuta este ejemplo, producirá el siguiente resultado:

En este capítulo, tomaremos un ejemplo para demostrar cómo dibujar un cubo 3D que se puede rotar usando los controles del mouse.

Ejemplo: dibujar un cubo interactivo

El siguiente programa muestra cómo rotar un cubo usando los controles del mouse:

<!doctype html>
<html>
   <body>
      <canvas width = "570" height = "570" id = "my_Canvas"></canvas>

      <script>
         /*============= Creating a canvas ======================*/
         var canvas = document.getElementById('my_Canvas');
         gl = canvas.getContext('experimental-webgl');

         /*========== Defining and storing the geometry ==========*/

         var vertices = [
            -1,-1,-1, 1,-1,-1, 1, 1,-1, -1, 1,-1,
            -1,-1, 1, 1,-1, 1, 1, 1, 1, -1, 1, 1,
            -1,-1,-1, -1, 1,-1, -1, 1, 1, -1,-1, 1,
            1,-1,-1, 1, 1,-1, 1, 1, 1, 1,-1, 1,
            -1,-1,-1, -1,-1, 1, 1,-1, 1, 1,-1,-1,
            -1, 1,-1, -1, 1, 1, 1, 1, 1, 1, 1,-1, 
         ];

         var colors = [
            5,3,7, 5,3,7, 5,3,7, 5,3,7,
            1,1,3, 1,1,3, 1,1,3, 1,1,3,
            0,0,1, 0,0,1, 0,0,1, 0,0,1,
            1,0,0, 1,0,0, 1,0,0, 1,0,0,
            1,1,0, 1,1,0, 1,1,0, 1,1,0,
            0,1,0, 0,1,0, 0,1,0, 0,1,0 
         ];

         var indices = [
            0,1,2, 0,2,3, 4,5,6, 4,6,7,
            8,9,10, 8,10,11, 12,13,14, 12,14,15,
            16,17,18, 16,18,19, 20,21,22, 20,22,23 
         ];

         // Create and store data into vertex buffer
         var vertex_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(vertices), gl.STATIC_DRAW);

         // Create and store data into color buffer
         var color_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(colors), gl.STATIC_DRAW);

         // Create and store data into index buffer
         var index_buffer = gl.createBuffer ();
         gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
         gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array(indices), gl.STATIC_DRAW);

         /*=================== SHADERS =================== */

         var vertCode = 'attribute vec3 position;'+
            'uniform mat4 Pmatrix;'+
            'uniform mat4 Vmatrix;'+
            'uniform mat4 Mmatrix;'+
            'attribute vec3 color;'+//the color of the point
            'varying vec3 vColor;'+
            'void main(void) { '+//pre-built function
               'gl_Position = Pmatrix*Vmatrix*Mmatrix*vec4(position, 1.);'+
               'vColor = color;'+
            '}';

         var fragCode = 'precision mediump float;'+
            'varying vec3 vColor;'+
            'void main(void) {'+
               'gl_FragColor = vec4(vColor, 1.);'+
            '}';

         var vertShader = gl.createShader(gl.VERTEX_SHADER);
         gl.shaderSource(vertShader, vertCode);
         gl.compileShader(vertShader);

         var fragShader = gl.createShader(gl.FRAGMENT_SHADER);
         gl.shaderSource(fragShader, fragCode);
         gl.compileShader(fragShader);

         var shaderprogram = gl.createProgram();
         gl.attachShader(shaderprogram, vertShader);
         gl.attachShader(shaderprogram, fragShader);
         gl.linkProgram(shaderprogram);

         /*======== Associating attributes to vertex shader =====*/
         var _Pmatrix = gl.getUniformLocation(shaderprogram, "Pmatrix");
         var _Vmatrix = gl.getUniformLocation(shaderprogram, "Vmatrix");
         var _Mmatrix = gl.getUniformLocation(shaderprogram, "Mmatrix");

         gl.bindBuffer(gl.ARRAY_BUFFER, vertex_buffer);
         var _position = gl.getAttribLocation(shaderprogram, "position");
         gl.vertexAttribPointer(_position, 3, gl.FLOAT, false,0,0);
         gl.enableVertexAttribArray(_position);

         gl.bindBuffer(gl.ARRAY_BUFFER, color_buffer);
         var _color = gl.getAttribLocation(shaderprogram, "color");
         gl.vertexAttribPointer(_color, 3, gl.FLOAT, false,0,0) ;
         gl.enableVertexAttribArray(_color);
         gl.useProgram(shaderprogram);

         /*==================== MATRIX ====================== */

         function get_projection(angle, a, zMin, zMax) {
            var ang = Math.tan((angle*.5)*Math.PI/180);//angle*.5
            return [
               0.5/ang, 0 , 0, 0,
               0, 0.5*a/ang, 0, 0,
               0, 0, -(zMax+zMin)/(zMax-zMin), -1,
               0, 0, (-2*zMax*zMin)/(zMax-zMin), 0 
			   ];
         }

         var proj_matrix = get_projection(40, canvas.width/canvas.height, 1, 100);
         var mo_matrix = [ 1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1 ];
         var view_matrix = [ 1,0,0,0, 0,1,0,0, 0,0,1,0, 0,0,0,1 ];

         view_matrix[14] = view_matrix[14]-6;

         /*================= Mouse events ======================*/

         var AMORTIZATION = 0.95;
         var drag = false;
         var old_x, old_y;
         var dX = 0, dY = 0;

         var mouseDown = function(e) {
            drag = true;
            old_x = e.pageX, old_y = e.pageY;
            e.preventDefault();
            return false;
         };

         var mouseUp = function(e){
            drag = false;
         };

         var mouseMove = function(e) {
            if (!drag) return false;
            dX = (e.pageX-old_x)*2*Math.PI/canvas.width,
            dY = (e.pageY-old_y)*2*Math.PI/canvas.height;
            THETA+= dX;
            PHI+=dY;
            old_x = e.pageX, old_y = e.pageY;
            e.preventDefault();
         };

         canvas.addEventListener("mousedown", mouseDown, false);
         canvas.addEventListener("mouseup", mouseUp, false);
         canvas.addEventListener("mouseout", mouseUp, false);
         canvas.addEventListener("mousemove", mouseMove, false);

         /*=========================rotation================*/

         function rotateX(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv1 = m[1], mv5 = m[5], mv9 = m[9];

            m[1] = m[1]*c-m[2]*s;
            m[5] = m[5]*c-m[6]*s;
            m[9] = m[9]*c-m[10]*s;

            m[2] = m[2]*c+mv1*s;
            m[6] = m[6]*c+mv5*s;
            m[10] = m[10]*c+mv9*s;
         }

         function rotateY(m, angle) {
            var c = Math.cos(angle);
            var s = Math.sin(angle);
            var mv0 = m[0], mv4 = m[4], mv8 = m[8];

            m[0] = c*m[0]+s*m[2];
            m[4] = c*m[4]+s*m[6];
            m[8] = c*m[8]+s*m[10];

            m[2] = c*m[2]-s*mv0;
            m[6] = c*m[6]-s*mv4;
            m[10] = c*m[10]-s*mv8;
         }

         /*=================== Drawing =================== */

         var THETA = 0,
         PHI = 0;
         var time_old = 0;

         var animate = function(time) {
            var dt = time-time_old;

            if (!drag) {
               dX *= AMORTIZATION, dY*=AMORTIZATION;
               THETA+=dX, PHI+=dY;
            }

            //set model matrix to I4

            mo_matrix[0] = 1, mo_matrix[1] = 0, mo_matrix[2] = 0,
            mo_matrix[3] = 0,

            mo_matrix[4] = 0, mo_matrix[5] = 1, mo_matrix[6] = 0,
            mo_matrix[7] = 0,

            mo_matrix[8] = 0, mo_matrix[9] = 0, mo_matrix[10] = 1,
            mo_matrix[11] = 0,

            mo_matrix[12] = 0, mo_matrix[13] = 0, mo_matrix[14] = 0,
            mo_matrix[15] = 1;

            rotateY(mo_matrix, THETA);
            rotateX(mo_matrix, PHI);

            time_old = time; 
            gl.enable(gl.DEPTH_TEST);

            // gl.depthFunc(gl.LEQUAL);

            gl.clearColor(0.5, 0.5, 0.5, 0.9);
            gl.clearDepth(1.0);
            gl.viewport(0.0, 0.0, canvas.width, canvas.height);
            gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

            gl.uniformMatrix4fv(_Pmatrix, false, proj_matrix);
            gl.uniformMatrix4fv(_Vmatrix, false, view_matrix);
            gl.uniformMatrix4fv(_Mmatrix, false, mo_matrix);

            gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, index_buffer);
            gl.drawElements(gl.TRIANGLES, indices.length, gl.UNSIGNED_SHORT, 0);

            window.requestAnimationFrame(animate);
         }
         animate(0);
      </script>
   </body>
</html>

Si ejecuta este ejemplo, producirá el siguiente resultado: