patriciogonzalezvivo / glslCanvas

Simple tool to load GLSL shaders on HTML Canvas using WebGL
http://patriciogonzalezvivo.github.io/glslCanvas/
MIT License
1.98k stars 183 forks source link

are `u_tex` working right with videos? #90

Open konsumer opened 2 months ago

konsumer commented 2 months ago

I have a shader like this:

#ifdef GL_ES
precision mediump float;
#endif

uniform vec2 u_resolution;

uniform sampler2D u_tex0; // britney.mp4
uniform sampler2D u_tex1; // hypnocat.png

void main() {
    vec2 uv = gl_FragCoord.xy / u_resolution.xy;
    gl_FragColor = texture2D(u_tex1, uv);
}

I am working on a live-editor with uniform-inputs.

Screenshot 2024-09-11 at 2 48 28 PM

Initially, britney.mp4 does not load, but after calling update() (with new shader, from prisma-live editor) it's in u_tex1 (where I would expect hypnocat.)

It seems like it has some sort of video support, but it's putting it in the wrong uniform.

If I set it up like this, initially, it works as expected:

uniform sampler2D u_tex0; // hypnocat.png
uniform sampler2D u_tex1; // https://prismjs.com/assets/img/spectrum.png

Possibly related:

Maybe 2 features I would like to add is working video-support and WEBCAM as URL. Is there interest in a PR?

konsumer commented 2 months ago

Additionally, I tried injecting a base64 URL (on user file-browse) and it does similar mixed-up texture buffers.

#ifdef GL_ES
precision mediump float;
#endif

uniform vec2 u_resolution;

uniform sampler2D u_tex0; // 
uniform sampler2D u_tex1; // https://prismjs.com/assets/img/spectrum.png

// these are input-knobs (0.0 - 1.0)
uniform float u_x0;
uniform float u_x1;
uniform float u_x2;
uniform float u_x3;

void main() {
  // scaled pixel-position
  vec2 uv = gl_FragCoord.xy / u_resolution.xy;

  // input 1 is mix-fader for tex0/1
  gl_FragColor = mix(texture2D(u_tex0, uv), texture2D(u_tex1, uv), u_x0);
}

No error, but the texture is not updated (even though other shader code is.)

If I set the initial shader to that, it also doesn't seem to use it (loads 2 copies of spectrum image.)

Update: I managed to get it sort of working with images, with this:

function setTex (index, url) {
  const r = new RegExp(`^[ \t]*uniform *sampler2D *u_tex${index} *;(.*)`, 'gm')
  const m = r.exec($code.value)
  const newline = `uniform sampler2D u_tex${index}; // ${url}`
  if (m && m[0]) {
    $code.value = $code.value.replace(r, newline)
    $code.dispatchEvent(new Event('input'))
    preview.loadTexture(`u_tex${index}`, url, {})
  } else {
    $code.value = `${newline}\n\n${$code.value}`
    $code.dispatchEvent(new Event('input'))
    preview.loadTexture(`u_tex${index}`, url, {})
  }
}

Basically, I need to update the shader first (adding the base64 URL comment, for code-sharing) and then force it to update texture with preview.loadTexture.

Initial base64-url comment is not working still, and videos still fail, and they are still swapped around, but this gets it loading them.

I also switched to objectURLs over base64, to make the code lighter:

// this works initially
uniform sampler2D u_tex0; // hypnocat.png
uniform sampler2D u_tex1; // https://prismjs.com/assets/img/spectrum.png

// this sort of works
// but hypnocat moved to u_tex0, and new objectURL is on u_tex1
uniform sampler2D u_tex0; // blob:http://localhost:5173/81e99c3d-046a-4ca7-8b34-b71887138af0
uniform sampler2D u_tex1; // https://prismjs.com/assets/img/spectrum.png
konsumer commented 2 months ago

I think the buggy image stuff is actually related to video. It bugs out the texture buffers, and swaps them around, but if I stick to just uploaded images, it seems fine.

Screenshot 2024-09-11 at 5 55 08 PM

Here the first image is the skull thing, and the second is the clown thing, so these are correct, so i think the issue is just videos.

You can play with it here, if you want to.

konsumer commented 2 months ago

I ended up just making my own much simpler glsl-canvas with threejs. Seems to work really well.

For video I am doing this:

  loadTexture (name, url, type) {
    if (type.startsWith('image')) {
      this.material.uniforms[name].value = new THREE.TextureLoader().load(url)
    } else if (type.startsWith('video')) {
      const video = document.createElement('video')
      video.src = url
      video.muted = true
      video.loop = true
      video.play()
      this.material.uniforms[name].value = new THREE.VideoTexture(video)
    }
  }