blob: 06b9a0d482cfcf1d272b042300fe4ee6090f537f (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
|
# Ray marching in Rust :rust: !
I'm currently learning Rust for fun and re-writing my [ascii-renderer](https://github.com/eug-vs/ascii-3d-renderer) from Python :python:. I'm using Ray Marching this time, here's how it looks (if you are on mobile, use desktop version of the site):
[![asciicast](https://asciinema.org/a/jsDqGeMkRwLeYhVIMqNbaec0M.svg)](https://asciinema.org/a/jsDqGeMkRwLeYhVIMqNbaec0M)
You can see that Ray Marching allows for some cool stuff like smooth surface blending and proper shadowing.
# Ray marching
Usually 3d renderers use a triangular mesh to describe objects in a scene. In Python version of `ascii-renderer` I defined each object as a set of points
(i.e object is defined by a function `__contains__` that determines whether the given point is in this object). It allowed me for some cool Ray Tracing stuff, but that was just me toying around.
Behold - Ray Marching! It's a cool rendering technique where you describe your scene with a *Distance Field* - each point in this field contains a distance from that point to the scene.
Well, actually it's a *Signed Distance Field* - the distance to the object is considered negative *inside* the object, and positive *outside*.
What's cool about Ray Marching is that since the whole scene is defined with mathematical expression, I can apply any mathematical transformations to it. For example I can make an object wavy
using a `sin`, or make it repeat forever with `%` operator. Also, `SDF` allows for different ways to combine objects - I can find a `union`, `difference`, and `intersection` for free!
## Performance
Currently it's rendering 4 shapes at ~19 FPS, which is pretty bad. That's probably due to my dirty and inefficient code :laughing: - I'm nowhere near a good understanding of Rust :rust: patterns.
My goal is to render 8 shapes at 24 FPS:
- 24 FPS is OK for a human eye
- 8 distinct shapes is usually enough to create a complex scene (ray marching allows for cool tricks that can multiply amount of your shapes without performance decrease)
# Plans
- Cleanup the code and increase performance
- Build an actually usable API
- Use `ncurses` instead of just printing to `STDOUT`
- Create an `ncurses` GUI for building a scene
- Allow importing/exporting scenes in `JSON`
- Do some magic :star: and expose API to the browser using `WebAssembly`
|