Home

Ray-Tracing: Rendering a Triangle

Distributed under the terms of the CC BY-NC-ND 4.0 License.

  1. Why Are Triangles Useful?
  2. Geometry of a Triangle
  3. Ray-Triangle Intersection: Geometric Solution
  4. Single vs Double Sided Triangle and Backface Culling
  5. Barycentric Coordinates
  6. Möller-Trumbore algorithm
  7. Source Code (external link GitHub)

Geometry of a Triangle

Reading time: 7 mins.

Basic Mathematical Tools

Before exploring the techniques to solve the ray-triangle intersection test, it's essential to familiarize ourselves with some foundational mathematical tools. As previously mentioned, a triangle's vertices define a plane. Each plane is characterized by a normal, which is a vector perpendicular to the plane's surface. Given the coordinates of the triangle's vertices \(A\), \(B\), and \(C\), we can compute vectors \(AB\) and \(AC\), which represent the edges from \(A\) to \(B\) and from \(A\) to \(C\), respectively. This is accomplished by subtracting \(B\) from \(A\) and \(C\) from \(A\).

A note here: Remember that Scratchapixel uses a right-hand coordinate system in all its lessons (with the up vector being the y-axis). Therefore, when you look at a triangle from the front, the triangle's vertices will be labeled \(v0\), \(v1\), and \(v2\) in a counter-clockwise fashion.

Similarly, it is logical to define the first vertex, from which subsequent vectors such as \(AB\) or \(AC\) will be calculated, as \(A\) or v0. However, choosing \(A\) over \(B\) or \(C\) is purely arbitrary. All the math we are about to provide will work exactly the same if you decide to start from vertex \(B\) or \(C\) (as shown in the figure below), though this may not seem like something you would typically do.

When we get to the lesson on Barycentric Coordinates, you will see that it might be convenient to consider a starting point other than \(A\). The only requirement, if you do so, is to ensure that you "walk" around the triangle in a counter-clockwise manner. For example, if you choose \(C\) as your starting vertex, the next vertex should be \(A\), followed by \(B\).

For now, and for the purposes of this chapter and the next, we will assume \(A\) or v0 as the first vertex from which subsequent vectors will be defined. However, keep in mind that this is not a strict requirement for the math to work.

To determine the plane's normal—which also serves as the triangle's normal, as the triangle resides within the plane—we calculate the cross product of vectors \(AB\) and \(AC\). Since these vectors lie in the same plane (connecting the triangle's vertices), the resulting vector from the cross product, denoted as \(N\), represents the sought-after normal (see Figure 1). The operations are summarized in the following pseudocode:

Vec3f v0(-1, -1, 0), v1(1, -1, 0), v2(0, 1, 0);
Vec3f e0 = v1 - v0; // Edge 0
Vec3f e1 = v2 - v0; // Edge 1
Vec3f N = cross(e0, e1); // Triangle's normal
normalize(N);

The mathematical representation of these operations is as follows:

$$ \begin{array}{l} AB = B - A = (1 - (-1), -1 - (-1), 0 - 0) = (2, 0, 0) \\ AC = C - A = (0 - (-1), 1 - (-1), 0 - 0) = (1, 2, 0) \\ N = AB \times AC \\ N_x = AB_y \cdot AC_z - AB_z \cdot AC_y = 0 \cdot 0 - 0 \cdot 2 = 0 \\ N_y = AB_z \cdot AC_x - AB_x \cdot AC_z = 0 \cdot 1 - 2 \cdot 0 = 0 \\ N_z = AB_x \cdot AC_y - AB_y \cdot AC_x = 2 \cdot 2 - 0 \cdot 1 = 4 \\ \end{array} $$

After normalizing \(N\), we obtain the vector \((0, 0, 1)\), which is parallel to the positive \(z\)-axis. This outcome is expected, given that the vertices \(A/v0\), \(B/v1\), and \(C/v2\) are positioned in the \(XY\)-plane.

Coordinate System Handedness

The order in which a triangle's vertices are defined significantly affects the orientation of the surface's normal. Defining vertices \(v_0\), \(v_1\), and \(v_2\) in a counter-clockwise (CCW) order results in a normal denoted by \(N\). Conversely, if the vertices were defined in a clockwise (CW) order, the normal obtained from the cross product of the two edges (\(e0 = v_1 - v_0\) and \(e1 = v_2 - v_0\)) would point in the opposite direction, yielding \(-N\). Similarly, if you compute \(e1 \times e0\) instead of \(e0 \times e1\), you would also get \(-N\). Therefore, be sure to use \(e0 \times e1\) when working with counter-clockwise ordered vertices, as expected when using a right-hand coordinate system.

Figure 1: Vectors \(e0\) and \(e1\) can be computed from \(v_1 - v_0\) and \(v_2 - v_0\), respectively. The normal of the triangle, denoted as \(N\), is the cross product of \(e0\) and \(e1\). The order in which the vertices are defined determines the direction of the normal. In this example, the vertices are arranged in a counter-clockwise manner.

When creating a triangle in Maya, which uses a right-hand coordinate system, generating the triangle's vertices in a CCW order implies that the x-axis (\(V_0V_1\)) becomes \(A\), the y-axis (\(V_0V_2\)) becomes \(B\), and \(C\), the cross product \(A \times B\), aligns parallel to the z-axis (Figure 2). However, if the vertices are created in a CW order, then \(C\) will point along the negative z-axis. Creating vertices in CW order and computing \(N\) from the cross product \(A \times B\) is equivalent to computing the cross product \(B \times A\) when vertices are created in CCW with \(A=V_1-V_0\) and \(B=V_2-V_0\).

Figure 2: Creating the vertices in CCW or CW changes the direction of the normal.

Imagine creating this triangle in a left-hand coordinate system, using the same vertices coordinates as in the right-hand coordinate system example (\(V_0 = (0,0,0)\), \(V_1=(1,0,0)\), and \(V_2=(0,1,0)\)). Following the same steps, we create vectors \(A\) (\(V_1-V_0\)) and \(B\) (\(V_2-V_0\)) and compute \(C\) (from \(A \times B\)). The result is \((0,0,1)\), identical to the right-hand coordinate system example, as the vertices' coordinates are the same. Hence, vectors \(A\) and \(B\), as well as the cross product \(A \times B\), are unchanged. By convention, the z-axis in the left-hand coordinate system points towards the screen (when the x-axis points to the right), aligning with our explanations on the handedness of coordinate systems. The cross product is an anti-commutative operation, as discussed in our lesson on Geometry.

Figure 3: Triangles created in left- or right-hand coordinate systems don't look the same when rendered from the same camera perspective (pointing down the negative z-axis).

The next step is to render each of these triangles from a camera pointing down the negative z-axis. The results of these tests, shown in Figure 4, reveal that the two images do not appear the same, even though the triangles' vertices share the same coordinates. This discrepancy is undesirable, as the choice of a coordinate system's handedness should not alter the geometry's appearance from the camera's viewpoint. Ideally, a rendering should consistently depict the same image of a triangle, regardless of the handedness used. This issue, while not directly related to ray-triangle intersection tests, poses a significant challenge. Typically, we address it by mirroring the camera along the negative z-axis, including both its position and orientation, as illustrated in Figure 4.

Figure 4: To achieve a consistent image of the triangle, it's necessary to mirror the camera about the z-axis and flip the triangle's normal.

However, this adjustment changes the direction of both the camera and the triangle's normal compared to their orientations before the mirroring. Initially, in Figure 3, the camera's direction and the triangle's normal are opposite. After the adjustment, they point in the same direction. These vectors—the triangle's normal and the camera direction—are crucial for shading; thus, their relative orientations remain unchanged. The solution for the flipped triangle issue involves mirroring the camera and reversing the normal's direction. This adjustment is only required if the triangle was modeled in a left-hand coordinate system application but rendered in a right-hand coordinate system renderer, or vice versa. For instance, creating geometry in Maya and rendering it in RenderMan, which uses right- and left-hand coordinate systems, respectively.

The specific order and direction in which vertices are defined is referred to as winding.

With these tools at our disposal, we are well-equipped to tackle the ray-triangle intersection test using simple geometry.

previousnext