Skip to content

Latest commit

 

History

History
94 lines (53 loc) · 6.46 KB

README.md

File metadata and controls

94 lines (53 loc) · 6.46 KB

Terminal Raymarching

Challenged to render 3D objects in the terminal; Hillarity ensues.

Milestones

Box Frame

Fully rendered distance map.


The pre-render

Succesfully calculated light value based on the normals and light strength(green), distance fog(red), and shadows(blue)


The Box As Intended

(Above) Object material detection.


Rotation

(Above) Object manipulation.


Spooky Hallway

(Above) Infinite pillairs.


Final

(Above) Fully furnished scene.


How it works

Canvas

Block character

So, initially the idea was to use full block character(█) as an overgrown pixel. The color was edited using ANSI codes. Typically, the color range would be limited, but it just so happened that st is a true color terminal. This means that I could set any RGB color with the \x1b[38;2;r;g;bm ANSI code.

// Image pending

Using a bit of vector magic I got the shader equivalent of "Hello world!" to work. And while it looked alright, I was bothered by the resolutions available. I could push canvas width all the way to 200 characters, but could only have 50 characters of height. A maximum resolution of 200x50 px; rather underwhelming.

However, something dawned on me while I was playing around with the screen dimensions. Typically, a character in terminal can have both its foreground and background color set. Background color would be useles with the full block character, since the character would simply cover it. But what if, there were a half block character; and it turns out there is(▄).

// Image pending

With this the screen height was doubled, and 200x100px was not looking bad at all.

Alas, up till this point, all that was achived was a static image. But with a simple loop more could be done. The only question was how to refresh the screen?

Animating

As I was doing this, a friend of mine(@cathalogue) suggested I check out the vim source code. Specifically, the term.c file. This file defines all the ANSI codes that make vim tick. After scrolling for a while, I managed to find a working clear screen code. I put all my code into a loop and made it clear the screen after each render, and... the screen tear was readly apparent.

It occured to me that maybe calling printf() to print a larger chunk of the screen less often was more performant than calling it on every single character. And so, I went ahead and buffered the output before the print. The results were undeniable. Screen tear was greatly reduced.

However, there was another problem, and that problem was due to clearing the screen. Because the drawing of the "pixels" went from left to right, top to bottom, the bottom right corrner would take more time to be reached by the rendering cursor. As a result, that area would take a bit longer to update. Clearing the screen makes this latency obvious; since it resets all the pixels, that area spends more time with no value.

So I went back to vim's term.c in search of a more fitting code. Then, I stumbled upon the "move cursor" ANSI code. If instead of clearing the screen, we zero out the cursor; return it to the begining, and then draw the next frame. Then, the troubled area would be a bit out dated every frame, but at least it would have a value. That way the redraw would not be as, if at all, noticable. By looking at the examples, you can see that I was correct.

Raymarching

RaymarchExample

Camera

So first we have to figure out the camera. Luckly, it is relatively simple. We pick our camera origin $\vec{co}$, and calculate our ray direction $\vec{rd}$. How do we calculate $\vec{rd}$? Well, the general idea of a camera is to collect the color value of all pixels within its Field of View(FOV), and project it onto a flat surace like our screen. In other words we take a vector $\vec{rd}$ that goes from the focal point to the position somewhere on a screen in front of it (like in the picture above), and we go in it's direction until it hits something.

Our screen has a width $w$, and a height $h$. We have to send out rays for every pixel on our screen and recieve a color value for our camera to work.

We can have two for loops go through all the values $V = \{ (x,\ y) \in \mathbb{N}^2\ \mid\ -\frac{w}{2} \le x \le \frac{w}{2},\ -\frac{h}{2} \le y \le \frac{h}{2} \}$.

We then normalize these values $\vec{v} \in V$, converting them in to the inteval of $[-1,\ 1]$, by dividing them like so: $\vec{v}\ * \ [\frac{w}{2},\ \frac{h}{2} ]$.

To get the ray that is going to shoot out of the camera point we merely plug these normalized values into a vector $\vec{rd} = [x,\ y,\ 1]$. Finally we normalize this vector i.e. make it so that it's magnitude is 1. This will be important later.

If we need to rotate our camera we merely need to multiply $\vec{rd}$ with a rotation matrix R, and then continue the steps below. The explaination of how to calculate a 3x3 rotation matrix can be found here.

Now we have a way to calculate $\vec{rd}$ for every pixel on screen, but how do we "go in it's direction until it hits something"? Notice first that this is actually two questions: "How to go in its direction", and: "When does it hit something?".

For the first question its rather simple. First we need to recall how $\vec{rd}$ is a unit vector. $1 * x = x$, and so too applies to a vector of magnitude 1, because, when a vector is multiplied by a scalar, only it's magnitude changes. So to get where the vector is suposed to be (lets call this $\vec{T}$, as in "Target") we simply multiply $\vec{rd}$ with the (distance traveled) scalar $d$: $\vec{T}\ =\ \vec{rd} * d$.

But, $\vec{T}$ starts of at the origin. Is that where we want to start off? We have to account for where the vector starts off from(recall $\vec{co}$). The completed formula looks like this: $\vec{T}\ =\ \vec{co} + \vec{rd} * d$.

Getting the distances

Now, lets tackle the second question: When does it hit something? Well, what does it mean to hit? To have $\vec{T}$ tangentialy touch the surface of a body? Maybe not so precise. It is a hit when $\vec{T}$ is marginally far away from the surface of the body.

Object Materials

Normal based shade

Light and Shadow