Índice

WebGL2Fundamentals.org

Fix, Fork, Contribute

Translação 2D WebGL2

Antes de passar para o 3D vamos ficar com 2D para um pouco mais. Vem comigo, por favor. Este artigo pode parecer extremamente óbvio para alguns, mas vou construir até um certo ponto em alguns artigos.

Este artigo é uma continuação de uma série que começa com os Fundamentos da WebGL2. Se você não os leu, sugiro que você leia pelo menos o primeiro, então volte aqui.

A translação é um nome de matemática extravagante que basicamente significa “mover” algo. Suponho que a mudança de uma frase do inglês para o japonês também se ajuste, mas neste caso estamos falando sobre a geometria em movimento. Usando o código de exemplo que fizemos no primeiro post você poderia facilmente transladar nosso retângulo apenas mudando os valores passados para setRectangle certo? Aqui está uma amostra baseada em nossa amostra anterior.

+  // Primeiro vamos fazer algumas variáveis
+  // para manter a translação, largura e altura do retângulo
+  var translation = [0, 0];
+  var width = 100;
+  var height = 30;
+  var color = [Math.random(), Math.random(), Math.random(), 1];
+
+  // Então vamos fazer uma função para
+  // recomeçar tudo. Podemos chamar essa
+  // função depois de atualizar a translação.

  // Desenhe uma cena.
  function drawScene() {
    webglUtils.resizeCanvasToDisplaySize(gl.canvas);

    // Diga a WebGL como converter do clip space para pixels
    gl.viewport(0, 0, gl.canvas.width, gl.canvas.height);

    // Limpe a tela
    gl.clearColor(0, 0, 0, 0);
    gl.clear(gl.COLOR_BUFFER_BIT | gl.DEPTH_BUFFER_BIT);

    // Diga para usar nosso programa (par de shaders)
    gl.useProgram(program);

    // Vincule o conjunto de atributos/buffers que queremos.
    gl.bindVertexArray(vao);

    // Passe na resolução da tela para que possamos converter de
    // pixels para clipspace no shader
    gl.uniform2f(resolutionUniformLocation, gl.canvas.width, gl.canvas.height);

    // Atualize o buffer de posição com posições do retângulo
    gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer);
*    setRectangle(gl, translation[0], translation[1], width, height);

    // Defina a cor.
    gl.uniform4fv(colorLocation, color);

    // Desenhe o retângulo.
    var primitiveType = gl.TRIANGLES;
    var offset = 0;
    var count = 6;
    gl.drawArrays(primitiveType, offset, count);
  }

No exemplo abaixo, anexei um par de controladores que atualizarão translation[0] e translation[1] e chama drawScene sempre que mudarem. Arraste os controladores para transladar o retângulo.

Por enquanto, tudo bem. Mas agora imagine que queríamos fazer o mesmo com uma forma mais complicada.

Digamos que queríamos desenhar um ‘F’ que consiste em 6 triângulos como este.

Bem, seguindo nosso código atual, teríamos que mudar o setRectangle para algo mais parecido com isso.

// Preencha o atual buffer ARRAY_BUFFER com os valores que definem uma letra 'F'.
function setGeometry(gl, x, y) {
  var width = 100;
  var height = 150;
  var thickness = 30;
  gl.bufferData(
      gl.ARRAY_BUFFER,
      new Float32Array([
          // left column
          x, y,
          x + thickness, y,
          x, y + height,
          x, y + height,
          x + thickness, y,
          x + thickness, y + height,

          // top rung
          x + thickness, y,
          x + width, y,
          x + thickness, y + thickness,
          x + thickness, y + thickness,
          x + width, y,
          x + width, y + thickness,

          // middle rung
          x + thickness, y + thickness * 2,
          x + width * 2 / 3, y + thickness * 2,
          x + thickness, y + thickness * 3,
          x + thickness, y + thickness * 3,
          x + width * 2 / 3, y + thickness * 2,
          x + width * 2 / 3, y + thickness * 3]),
      gl.STATIC_DRAW);
}

Você pode ver que não vai se dimensionar bem. Se quisermos desenhar uma geometria muito complexa com centenas ou milhares de linhas, teríamos que escrever um código bastante complexo. Além disso, cada vez que desenhamos, o JavaScript tem que atualizar todos os pontos.

Há uma maneira mais simples. Basta carregar a geometria e fazer a translação no shader.

Aqui está o novo shader

#version 300 es

// um atributo é um input (in) para um vertex shader.
// Ele receberá dados de um buffer
in vec4 a_position;

// Usado para passar na resolução da tela
uniform vec2 u_resolution;

+// translação para adicionar à posição
+uniform vec2 u_translation;

// todos os shaders têm uma função principal
void main() {
+  // Adicionar na translação
+  vec2 position = a_position + u_translation;

  // converta a posição de pixels de 0,0 a 1,0
*  vec2 zeroToOne = position / u_resolution;

  // converter de 0-> 1 para 0-> 2
  vec2 zeroToTwo = zeroToOne * 2.0;

  // converter de 0-> 2 para -1 -> + 1 (clipspace)
  vec2 clipSpace = zeroToTwo - 1.0;

  gl_Position = vec4(clipSpace * vec2(1, -1), 0, 1);
}

e vamos reestruturar o código um pouco. Por um lado, só precisa definir a geometria uma vez.

// Preencha o atual buffer ARRAY_BUFFER
// com os valores que definem uma letra 'F'.
function setGeometry(gl) {
  gl.bufferData(
      gl.ARRAY_BUFFER,
      new Float32Array([
          // coluna esquerda
          0, 0,
          30, 0,
          0, 150,
          0, 150,
          30, 0,
          30, 150,

          // primeiro degrau
          30, 0,
          100, 0,
          30, 30,
          30, 30,
          100, 0,
          100, 30,

          // segundo nível
          30, 60,
          67, 60,
          30, 90,
          30, 90,
          67, 60,
          67, 90]),
      gl.STATIC_DRAW);
}

Então precisamos atualizar u_translation antes de desenhar com a translação que desejamos.

  ...

+  var translationLocation = gl.getUniformLocation(
+             program, "u_translation");

  ...

+  // Definir geometria.
+  setGeometry(gl);

  ...

  // Desenhe a cena.
  function drawScene() {
    webglUtils.resizeCanvasToDisplaySize(gl.canvas);

    // Diga a WebGL como converter de clip space para pixels
    gl.viewport(0, 0, gl.canvas.width, gl.canvas.height);

    // Diga para usar nosso programa (par de shaders)
    gl.useProgram(program);

    // Vincule o conjunto de atributos/buffers que queremos.
    gl.bindVertexArray(vao);

    // Passe na resolução da tela para que possamos converter
    // pixels para clipspace no shader
    gl.uniform2f(resolutionUniformLocation, gl.canvas.width, gl.canvas.height);

    // Defina a cor.
    gl.uniform4fv(colorLocation, color);

+    // Defina a translação.
+    gl.uniform2fv(translationLocation, translation);

    // Desenhe o retângulo.
    var primitiveType = gl.TRIANGLES;
    var offset = 0;
*    var count = 18;
    gl.drawArrays(primitiveType, offset, count);
  }

Observe que setGeometry é chamado apenas uma vez. Já não está dentro do drawScene.

E aqui esse exemplo. Mais uma vez, arraste os controles deslizantes para atualizar a tradução.

Agora, quando desenhamos, a WebGL está fazendo praticamente tudo. Tudo o que estamos fazendo é estabelecer uma translação e pedir para desenhar. Mesmo que a nossa geometria tivesse dezenas de milhares de pontos, o código principal ficaria o mesmo.

Se você quiser, você pode comparar a versão usa o JavaScript complexo acima para atualizar todos os pontos.

Espero que este exemplo não seja muito óbvio. No próximo artigo, passaremos para a rotação.

Problemas/Bug? Pergunte no github.
comments powered by Disqus