WebGL + Rust: Basic Water Tutorial(chinedufn.com) |
WebGL + Rust: Basic Water Tutorial(chinedufn.com) |
One thing that can be done is playing around with the numbers / uniforms in the fragment shader to change the look. For many use cases you can get pretty far with that without actually changing the implementation in this demo.
If a more physically based level of realism is needed (like the water ripples on click in the demo that you linked) you can use techniques like height simulation based on a wave equation [1].
You'll of course need to balance realism with realtime feasibility - but often times getting a much better look comes down to tweaking some numbers until things look good!
[1] - https://developer.nvidia.com/gpugems/GPUGems/gpugems_ch01.ht...
> The way that the water appear to move comes from slightly altering the point on the refraction and reflection textures that we sample from every frame. > > These slight offsets comes from sampling a du/dv map, which is just a texture that encodes different x and y offsets.
The app in the link you posted does indeed look like it's solving at least a shallow water system of equations, and looks very excited to me as a wave physicist.
https://www.figma.com/blog/building-a-professional-design-to...
- https://www.youtube.com/watch?v=M6RLvGqQU10&list=PLgC1L0fKd7...
fn render_reflection_fbo(
&mut self,
gl: &WebGlRenderingContext,
state: &State,
assets: &Assets,
) {
gl.bind_framebuffer(...);
gl.viewport(...);
gl.clear_color(...);
gl.clear(...);
Setting the viewport and clear color sure look to be mutating, but they are being called on a shared reference (&WebGlRenderingContext).Why is it designed this way? Is this design considered sound, or is it a hack?
There's a recent issue [1] in the wasm-bindgen repo that explains this, but essentially:
- In this demo the WebGLRenderingContext was created on the Rust side (technically by calling a light JS shim), but it could've just as well have been passed in from JS.
- This means that JS could very easily have had access to the WebGLRenderingContext.
- This means that we cannot guarantee that it is immutable. Who is to say that there isn't a line of JS `gl.mutated! = true;` that we don't know about? We have no guarantees that whoever instantiated this WebAssembly module isn't doing something funky.
- So we treat most DOM / JS APIs as if they have interior mutability, so more or less you can think of them as `RefCell`'s [2]
---
As an aside. When you work with WebGL you're just making calls to the GPU and your state on the GPU gets mutated. The object just controls state (WebGlRenderingContext) does not get mutated (aside from maybe a couple things that I'm forgetting..?).
As in.. `gl.viewport` doesn't actually mutate `gl`. But again.. just a small aside!
In general the idea here is that we can't guarantee that there is only one mutable reference so there is no point in calling these things `&mut`, even for things that really do mutate.
---
So yup! Interior mutability without runtime checks for many of the APIs is "the right way" in today's idiomatic Rust + WASM + DOM access since you can't guarantee that there isn't foul play going on from whoever instantiated the module.
---
Does this cause issues? It hasn't (noticeably) for me yet but I'm only a couple months into using `web_sys` so I haven't worked with every single DOM API.. so grain of salt!
[1] - https://github.com/rustwasm/wasm-bindgen/issues/1061
[2] - https://doc.rust-lang.org/std/cell/struct.RefCell.html
- I see that I poorly communicated this in the title, but I was actually shooting less for "Rust/WASM makes this better!!" and more for "Hey, this is possible with pretty much just Rust/WASM!" I ended up hand writing about 10 lines of JS [1] in total
- More minor point - WebGL commands don't need to be called via JS [2] after anyref and host bindings land, and I think that it's important for people to know that it's possible to do things on the web without JS (whether that's a good idea is very situational)!
So yeah apologies if the title could be a bit better - but the main goal was to share with others that it's possible to build 3d experiences on the web with Rust today!
And - just to not come off wrong here - I'm not saying whether this is right or wrong and when to go this route. I am just interested in demonstrating how to do it :)
But all in all you're right - all of this same experience could totally be built with only JS if someone so desired!
[1] - https://github.com/chinedufn/webgl-water-tutorial/blob/maste...
[2] - https://github.com/WebAssembly/reference-types/blob/master/p...
That said, it's unnecessarily dismissive. It's still a cool project! And a useful resource for anyone who does need to use Rust with WebGL for whatever reason.
You can skim through my papers at https://milancurcic.com/publications. There is also a nice short video where my boss describes what we do in the lab: https://www.youtube.com/watch?v=B6c20IqR2mw
Today WebGL is commonly referred to as a JS API because pretty much every WebGL app is written using JS.
But, even though that MDN link quite literally says "JavaScript API", it turns out that WebGL is entirely specified in WebIDL and isn't coupled to JS.
Because of this, `wasm-bindgen` [1] is able to auto-generate all of the WebGL bindings [2] that I used for this demo.
Today `wasm-bindgen` automatically generates some JS shims because that's the only way to call the WebGL APIs at this time, but once anyref and host bindings land you'll be able to interface with the WebGL APIs directly without going through JS.
`wasm-bindgen` already plans to just replace those JS shims with direct host bindings.
But yup - totally agree with your points here - just sharing this minor detail that will be more relevant in the future!
[1] - https://github.com/rustwasm/wasm-bindgen
[2] - https://rustwasm.github.io/wasm-bindgen/api/web_sys/struct.W...