Noitidart Noitidart - 3 months ago 11
Javascript Question

Load image, scale, then overlap - Badge an image

All my current code is in a worker. I now have the need for a simple graphics issue. I have two square images with known width/height. I need to take the first and draw it at 64x64, then take the second and draw it at 16x16 and position at the bottom right corner. The final image is 64x64. I am basically trying to make the 2nd image a badge on the first image.

So now this is piece of cake in 2d canvas, however I cannot (for some reasons) do communication with the doc, I have to do this all in canvas, in Firefox we have support since Firefox 44 (last year) for webgl canvas. So I am trying to get this done in that.

Here is what I put together from around the web, can you please help me complete it. My drawImage method is what needs to work properly I set it up to take arguments but I remove all my code that was respecting it, because it was really breaking things. I have to edit the

vec4
second arg for scaling (for instance
1.5
will make it scale to half the size), but i am not able to figure out how to position.



function doit() {
var img, tex, vloc, tloc, vertexBuff, texBuff;

var cvs3d = document.getElementById('cvs');
var ctx3d = cvs3d.getContext('experimental-webgl');
var uLoc;

// create shaders
var vertexShaderSrc =
'attribute vec2 aVertex;' +
'attribute vec2 aUV;' +
'varying vec2 vTex;' +
'uniform vec2 pos;' +
'void main(void) {' +
' gl_Position = vec4(aVertex + pos, 0.0, 1.0);' +
' vTex = aUV;' +
'}';

var fragmentShaderSrc =
'precision highp float;' +
'varying vec2 vTex;' +
'uniform sampler2D sampler0;' +
'void main(void){' +
' gl_FragColor = texture2D(sampler0, vTex);' +
'}';

var vertShaderObj = ctx3d.createShader(ctx3d.VERTEX_SHADER);
var fragShaderObj = ctx3d.createShader(ctx3d.FRAGMENT_SHADER);
ctx3d.shaderSource(vertShaderObj, vertexShaderSrc);
ctx3d.shaderSource(fragShaderObj, fragmentShaderSrc);
ctx3d.compileShader(vertShaderObj);
ctx3d.compileShader(fragShaderObj);

var progObj = ctx3d.createProgram();
ctx3d.attachShader(progObj, vertShaderObj);
ctx3d.attachShader(progObj, fragShaderObj);

ctx3d.linkProgram(progObj);
ctx3d.useProgram(progObj);

ctx3d.viewport(0, 0, 64, 64);

vertexBuff = ctx3d.createBuffer();
ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([-1, 1, -1, -1, 1, -1, 1, 1]), ctx3d.STATIC_DRAW);

texBuff = ctx3d.createBuffer();
ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([0, 1, 0, 0, 1, 0, 1, 1]), ctx3d.STATIC_DRAW);

vloc = ctx3d.getAttribLocation(progObj, 'aVertex');
tloc = ctx3d.getAttribLocation(progObj, 'aUV');
uLoc = ctx3d.getUniformLocation(progObj, 'pos');

var drawImage = function(imgobj, x, y, w, h) {
tex = ctx3d.createTexture();
ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MIN_FILTER, ctx3d.NEAREST);
ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MAG_FILTER, ctx3d.NEAREST);
ctx3d.texImage2D(ctx3d.TEXTURE_2D, 0, ctx3d.RGBA, ctx3d.RGBA, ctx3d.UNSIGNED_BYTE, imgobj);

ctx3d.enableVertexAttribArray(vloc);
ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
ctx3d.vertexAttribPointer(vloc, 2, ctx3d.FLOAT, false, 0, 0);

ctx3d.enableVertexAttribArray(tloc);
ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
ctx3d.vertexAttribPointer(tloc, 2, ctx3d.FLOAT, false, 0, 0);

ctx3d.drawArrays(ctx3d.TRIANGLE_FAN, 0, 4);
};

img = new Image();
img.src = '';

img.onload = function() {
console.log('drawing base image now');
drawImage(this, 0, 0, 64, 64);

var img2 = new Image();
img2.src = '';
img2.onload = function() {
drawImage(img2, 64-16, 64-16, 16, 16); // draw in bottom right corner
}
};
}

window.onload = function() {
doit();
}

#cvs {
border: 1px solid red;
}

<canvas id="cvs" width=64 height=64></canvas>




Answer

There's a million answers to this question because WebGL is a rasterization library

So for example

  1. You could adjust the viewport

  2. You could add a uniform vec2 scale on top of your pos since without a scale you can't make the quad smaller/larger

  3. You could update the vertices before drawing.

  4. You could multiply aVertex by a uniform mat3 matrix which would let you arbitrarily position, scale, and rotate

  5. You could multiply aVertex by a uniform mat4 matrix which would let you arbitrarily position, scale, rotate, and project into 3d

5 is this the standard solution because it's the most flexible. This article goes over using a mat3 and shows your method before it and why using a matrix is a far better choice, it then expands on that to a mat4 for doing full 3d.

So it's not clear what you want. Do you want to learn the "good way (#5) or do you just want your code to work with as few changes as possible.

Just for fun here's #1

function doit() {
  var img, tex, vloc, tloc, vertexBuff, texBuff;

  var cvs3d = document.getElementById('cvs');
  var ctx3d = cvs3d.getContext('experimental-webgl', { 
    preserveDrawingBuffer: true, 
  });
  var uLoc;

  // create shaders
  var vertexShaderSrc =
      'attribute vec2 aVertex;' +
      'attribute vec2 aUV;' +
      'varying vec2 vTex;' +
      'uniform vec2 pos;' +
      'void main(void) {' +
      '  gl_Position = vec4(aVertex + pos, 0.0, 1.0);' +
      '  vTex = aUV;' +
      '}';

  var fragmentShaderSrc =
      'precision highp float;' +
      'varying vec2 vTex;' +
      'uniform sampler2D sampler0;' +
      'void main(void){' +
      '  gl_FragColor = texture2D(sampler0, vTex);' +
      '}';

  var vertShaderObj = ctx3d.createShader(ctx3d.VERTEX_SHADER);
  var fragShaderObj = ctx3d.createShader(ctx3d.FRAGMENT_SHADER);
  ctx3d.shaderSource(vertShaderObj, vertexShaderSrc);
  ctx3d.shaderSource(fragShaderObj, fragmentShaderSrc);
  ctx3d.compileShader(vertShaderObj);
  ctx3d.compileShader(fragShaderObj);

  var progObj = ctx3d.createProgram();
  ctx3d.attachShader(progObj, vertShaderObj);
  ctx3d.attachShader(progObj, fragShaderObj);

  ctx3d.linkProgram(progObj);
  ctx3d.useProgram(progObj);

  vertexBuff = ctx3d.createBuffer();
  ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
  ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([-1, 1, -1, -1, 1, -1, 1, 1]), ctx3d.STATIC_DRAW);

  texBuff = ctx3d.createBuffer();
  ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
  ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([0, 1, 0, 0, 1, 0, 1, 1]), ctx3d.STATIC_DRAW);

  vloc = ctx3d.getAttribLocation(progObj, 'aVertex');
  tloc = ctx3d.getAttribLocation(progObj, 'aUV');
  uLoc = ctx3d.getUniformLocation(progObj, 'pos');

  var drawImage = function(imgobj, x, y, w, h) {
    tex = ctx3d.createTexture();
    ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
    ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MIN_FILTER, ctx3d.NEAREST);
    ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MAG_FILTER, ctx3d.NEAREST);
    ctx3d.texImage2D(ctx3d.TEXTURE_2D, 0, ctx3d.RGBA, ctx3d.RGBA, ctx3d.UNSIGNED_BYTE, imgobj);

    ctx3d.enableVertexAttribArray(vloc);
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
    ctx3d.vertexAttribPointer(vloc, 2, ctx3d.FLOAT, false, 0, 0);

    ctx3d.enableVertexAttribArray(tloc);
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
    ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
    ctx3d.vertexAttribPointer(tloc, 2, ctx3d.FLOAT, false, 0, 0);

    ctx3d.viewport(x, y, w, h);
    ctx3d.drawArrays(ctx3d.TRIANGLE_FAN, 0, 4);
  };

  img = new Image();
  img.src = '';

  img.onload = function() {
    console.log('drawing base image now');
    drawImage(this, 0, 0, 64, 64);

    var img2 = new Image();
    img2.src = '';
    img2.onload = function() {
      drawImage(img2, 64-16, 64-16, 16, 16); // draw in bottom right corner
    }
  };
}

window.onload = function() {
  doit();
}
#cvs {
  border: 1px solid red;
}
<canvas id="cvs" width=64 height=64></canvas>

Note that WebGL, by default, clears the canvas the next time you render so the fact that you're rendering twice, once after each image loads, means you're only going to end up with the 2nd result. To prevent that you need to pass { preserveDrawingBuffer: true, } as the second parameter to getContext.

I gotta be honest, this feels like a "do my homework for me question". Like did you even look up a single article on WebGL? I suppose I should give you the benefit of the doubt but there's so much here.

Do you know WebGL only cares about clip space coordinates?

Do you know WebGL -1 is at the bottom and +1 is at the top?

Do you know WebGL can't draw non-power of 2 images unless you set various texture parameters? Your example works because the image is 64x64 but if it was 65x64 it would fail.

So that brings up more questions.

The textures are upside down. Again, it's up to you to decide how to fix them.

You could

  1. Flip your texture coordinates

  2. Load the texture flipped with gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, true);

  3. Use a negative scale (assuming you go with solution #2 above). That will complicate your pos settings

  4. Adjust your texture coordinates in your shader

  5. Negate gl_Position.y in your shader

  6. Use solutions #4 (mat3) or #5 (mat4) above and create a projection matrix makes flips the space.

Again the matrix solutins are the best solutions and you should really go read the articles I posted.

That said, hacking your current code in the #2 solution

function doit() {
  var img, tex, vloc, tloc, sloc, vertexBuff, texBuff;

  var cvs3d = document.getElementById('cvs');
  var ctx3d = cvs3d.getContext('experimental-webgl', { 
    preserveDrawingBuffer: true, 
  });
  var uLoc;

  // create shaders
  var vertexShaderSrc = `
      attribute vec2 aVertex;
      attribute vec2 aUV;
      varying vec2 vTex;
      uniform vec2 pos;
      uniform vec2 scale; 
      void main(void) {
        gl_Position = vec4(aVertex * scale + pos, 0.0, 1.0);
        vTex = aUV;
      }`;

  var fragmentShaderSrc = `
      precision highp float;
      varying vec2 vTex;
      uniform sampler2D sampler0;
      void main(void){
        gl_FragColor = texture2D(sampler0, vTex);
      }`;

  var vertShaderObj = ctx3d.createShader(ctx3d.VERTEX_SHADER);
  var fragShaderObj = ctx3d.createShader(ctx3d.FRAGMENT_SHADER);
  ctx3d.shaderSource(vertShaderObj, vertexShaderSrc);
  ctx3d.shaderSource(fragShaderObj, fragmentShaderSrc);
  ctx3d.compileShader(vertShaderObj);
  ctx3d.compileShader(fragShaderObj);

  var progObj = ctx3d.createProgram();
  ctx3d.attachShader(progObj, vertShaderObj);
  ctx3d.attachShader(progObj, fragShaderObj);

  ctx3d.linkProgram(progObj);
  ctx3d.useProgram(progObj);

  vertexBuff = ctx3d.createBuffer();
  ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
  ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([-1, 1, -1, -1, 1, -1, 1, 1]), ctx3d.STATIC_DRAW);

  texBuff = ctx3d.createBuffer();
  ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
  ctx3d.bufferData(ctx3d.ARRAY_BUFFER, new Float32Array([0, 1, 0, 0, 1, 0, 1, 1]), ctx3d.STATIC_DRAW);

  vloc = ctx3d.getAttribLocation(progObj, 'aVertex');
  tloc = ctx3d.getAttribLocation(progObj, 'aUV');
  uLoc = ctx3d.getUniformLocation(progObj, 'pos');
  sLoc = ctx3d.getUniformLocation(progObj, 'scale');

  var drawImage = function(imgobj, x, y, w, h) {
    tex = ctx3d.createTexture();
    ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
    ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MIN_FILTER, ctx3d.NEAREST);
    ctx3d.texParameteri(ctx3d.TEXTURE_2D, ctx3d.TEXTURE_MAG_FILTER, ctx3d.NEAREST);
    ctx3d.texImage2D(ctx3d.TEXTURE_2D, 0, ctx3d.RGBA, ctx3d.RGBA, ctx3d.UNSIGNED_BYTE, imgobj);

    ctx3d.enableVertexAttribArray(vloc);
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, vertexBuff);
    ctx3d.vertexAttribPointer(vloc, 2, ctx3d.FLOAT, false, 0, 0);

    ctx3d.enableVertexAttribArray(tloc);
    ctx3d.bindBuffer(ctx3d.ARRAY_BUFFER, texBuff);
    ctx3d.bindTexture(ctx3d.TEXTURE_2D, tex);
    ctx3d.vertexAttribPointer(tloc, 2, ctx3d.FLOAT, false, 0, 0);
    
    // convert x, y to clip space (assuming viewport matches canvas size)
    var cx = x / ctx3d.canvas.width * 2 - 1;
    var cy = y / ctx3d.canvas.height * 2 - 1;
    
    // convert w, h to clip space (quad is 2 units big)
    var cw = w / ctx3d.canvas.width;
    var ch = h / ctx3d.canvas.height;
    
    // because the quad centered over 0.0 we have to add in 
    // half the width and height (cw, ch are already half because
    // it's 2 unit quad
    cx += cw;
    cy += ch;
    
    // then we negate cy and ch because webgl -1 is at the bottom
    ctx3d.uniform2f(uLoc, cx, -cy)
    ctx3d.uniform2f(sLoc, cw, -ch);

    ctx3d.drawArrays(ctx3d.TRIANGLE_FAN, 0, 4);
  };

  img = new Image();
  img.src = '';

  img.onload = function() {
    console.log('drawing base image now');
    drawImage(this, 0, 0, 64, 64);

    var img2 = new Image();
    img2.src = '';
    img2.onload = function() {
      drawImage(img2, 64-16, 64-16, 16, 16); // draw in bottom right corner
    }
  };
}

doit();
#cvs {
  border: 1px solid red;
}
<canvas id="cvs" width=64 height=64></canvas>

Personally I'd suggest you read up on WebGL including how to implement drawImage and how to create a matrix stack

PS: pngcrush is your friend