are `u_tex` working right with videos?
konsumer opened this issue · 3 comments
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.
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?
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
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.
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.
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)
}
}