WebGL - Traduzione
Finora, abbiamo discusso come disegnare varie forme e applicare i colori in esse utilizzando WebGL. Qui, in questo capitolo, faremo un esempio per mostrare come tradurre un triangolo.
Traduzione
La traduzione è uno dei affine transformationsfornito da WebGL. Usando la traduzione, possiamo spostare un triangolo (qualsiasi oggetto) sul piano xyz. Supponiamo di avere un triangolo [a, b, c] e di voler spostare il triangolo in una posizione che è di 5 unità verso l'asse X positivo e 3 unità verso l'asse Y positivo. Allora i nuovi vertici sarebbero [a + 5, b + 3, c + 0]. Ciò significa che per tradurre il triangolo, dobbiamo aggiungere le distanze di traslazione, ad esempio, tx, ty, tz a ciascun vertice.
Poiché è un file per-vertex operation, possiamo portarlo nel programma vertex shader.
Nel vertex shader, insieme all'attributo, coordinates(che mantengono le posizioni dei vertici), definiamo una variabile uniforme che contiene le distanze di traslazione (x, y, z). Successivamente, aggiungiamo questa variabile uniforme alla variabile coordinate e assegniamo il risultato al filegl_Position variabile.
Note - Poiché il vertex shader verrà eseguito su ogni vertice, tutti i vertici del triangolo verranno tradotti.
Passaggi per tradurre un triangolo
I seguenti passaggi sono necessari per creare un'applicazione WebGL per disegnare un triangolo e quindi tradurlo in una nuova posizione.
Step 1 − Prepare the Canvas and Get the WebGL Rendering Context
In questo passaggio, otteniamo l'oggetto contesto di rendering WebGL utilizzando getContext().
Step 2 − Define the Geometry and Store it in the Buffer Objects
Dato che stiamo disegnando un triangolo, dobbiamo passare tre vertici del triangolo e memorizzarli nei buffer.
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
In questo passaggio, è necessario scrivere i programmi vertex shader e fragment shader, compilarli e creare un programma combinato collegando questi due programmi.
Vertex Shader- Nel vertex shader del programma, definiamo un attributo vettoriale per memorizzare le coordinate 3D. Insieme ad esso, definiamo una variabile uniforme per memorizzare le distanze di traslazione e, infine, aggiungiamo questi due valori e lo assegniamo agl_position che mantiene la posizione finale dei vertici.
var vertCode =
'attribute vec4 coordinates;' +
'uniform vec4 translation;'+
'void main(void) {' +
' gl_Position = coordinates + translation;' +
'}';
Fragment Shader - Nello shader del frammento, assegniamo semplicemente il colore del frammento alla variabile 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
In questo passaggio, associamo gli oggetti buffer al programma shader.
Step 5 − Drawing the Required Object
Dato che stiamo disegnando il triangolo usando gli indici, useremo il metodo drawArrays(). A questo metodo dobbiamo passare il numero di vertici / elementi da considerare. Dato che stiamo disegnando un triangolo, passeremo 3 come parametro.
gl.drawArrays(gl.TRIANGLES, 0, 3);
Esempio: tradurre un triangolo
L'esempio seguente mostra come tradurre un triangolo sul piano 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>
Se esegui questo esempio, produrrà il seguente output: