Drawing an image using WebGL

后端 未结 2 365
南笙
南笙 2021-01-02 04:04

Please could anyone explain how to draw an image on a WebGL canvas? At the moment, on a regular \'2d\' canvas, I\'m using this:

var canvas = document.getElem         


        
2条回答
  •  借酒劲吻你
    2021-01-02 04:10

    If it was up to me I'd do it with a unit quad and a matrix like this

    Given these shaders

    vertex shader

    attribute vec2 a_position;
    
    uniform mat3 u_matrix;
    
    varying vec2 v_texCoord;
    
    void main() {
       gl_Position = vec4(u_matrix * vec3(a_position, 1), 1);
    
       // because we're using a unit quad we can just use
       // the same data for our texcoords.
       v_texCoord = a_position;  
    }
    

    fragment shader

    precision mediump float;
    
    // our texture
    uniform sampler2D u_image;
    
    // the texCoords passed in from the vertex shader.
    varying vec2 v_texCoord;
    
    void main() {
       gl_FragColor = texture2D(u_image, v_texCoord);
    }
    
    

    I'd create a unit quad and then fill out a 3x3 matrix to translate, rotate, and scale it where I needed it to be

      var dstX = 20;
      var dstY = 30;
      var dstWidth = 64;
      var dstHeight = 64;
    
      // convert dst pixel coords to clipspace coords      
      var clipX = dstX / gl.canvas.width  *  2 - 1;
      var clipY = dstY / gl.canvas.height * -2 + 1;
      var clipWidth = dstWidth  / gl.canvas.width  *  2;
      var clipHeight = dstHeight / gl.canvas.height * -2;
    
      // build a matrix that will stretch our
      // unit quad to our desired size and location
      gl.uniformMatrix3fv(u_matrixLoc, false, [
          clipWidth, 0, 0,
          0, clipHeight, 0,
          clipX, clipY, 1,
        ]);
    

    "use strict";
    
    window.onload = main;
    
    function main() {
      var image = new Image();
      // using a dataURL because stackoverflow
      image.src = "";  // MUST BE SAME DOMAIN!!!
      image.onload = function() {
        render(image);
      }
    }
    
    function render(image) {
      // Get A WebGL context
      var canvas = document.getElementById("c");
      var gl = canvas.getContext("webgl");
      if (!gl) {
        return;
      }
    
      // setup GLSL program
      var program = webglUtils.createProgramFromScripts(gl, ["2d-vertex-shader", "2d-fragment-shader"]);
      gl.useProgram(program);
    
      // look up where the vertex data needs to go.
      var positionLocation = gl.getAttribLocation(program, "a_position"); 
      
      // look up uniform locations
      var u_imageLoc = gl.getUniformLocation(program, "u_image");
      var u_matrixLoc = gl.getUniformLocation(program, "u_matrix");
    
      // provide texture coordinates for the rectangle.
      var positionBuffer = gl.createBuffer();
      gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer);
      gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([
          0.0,  0.0,
          1.0,  0.0,
          0.0,  1.0,
          0.0,  1.0,
          1.0,  0.0,
          1.0,  1.0]), gl.STATIC_DRAW);
      gl.enableVertexAttribArray(positionLocation);
      gl.vertexAttribPointer(positionLocation, 2, gl.FLOAT, false, 0, 0);
    
      var texture = gl.createTexture();
      gl.bindTexture(gl.TEXTURE_2D, texture);
    
      // Set the parameters so we can render any size image.
      gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
      gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
      gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
      gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
    
      // Upload the image into the texture.
      gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, image);
    
      var dstX = 20;
      var dstY = 30;
      var dstWidth = 64;
      var dstHeight = 64;
    
      // convert dst pixel coords to clipspace coords      
      var clipX = dstX / gl.canvas.width  *  2 - 1;
      var clipY = dstY / gl.canvas.height * -2 + 1;
      var clipWidth = dstWidth  / gl.canvas.width  *  2;
      var clipHeight = dstHeight / gl.canvas.height * -2;
    
      // build a matrix that will stretch our
      // unit quad to our desired size and location
      gl.uniformMatrix3fv(u_matrixLoc, false, [
          clipWidth, 0, 0,
          0, clipHeight, 0,
          clipX, clipY, 1,
        ]);
    
      // Draw the rectangle.
      gl.drawArrays(gl.TRIANGLES, 0, 6);
    }
    canvas { 
      border: 1px solid black;
    }
    
      
    
    
    
    

    Here's some articles that will explain the the matrix math

提交回复
热议问题