Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

are u_tex working right with videos? #90

Open
konsumer opened this issue Sep 11, 2024 · 3 comments
Open

are u_tex working right with videos? #90

konsumer opened this issue Sep 11, 2024 · 3 comments

Comments

@konsumer
Copy link

konsumer commented Sep 11, 2024

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 konsumer changed the title are u_tex working right? are u_tex working right with videos? Sep 11, 2024
@konsumer
Copy link
Author

konsumer commented Sep 12, 2024

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
Copy link
Author

konsumer commented Sep 12, 2024

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
Copy link
Author

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)
    }
  }

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

1 participant