Computer Graphics (CS 4300) 2010S: Lecture 9
Today
- triangle mesh data structures
- barycentric coordinates
- rasterizing triangles
Triangle Mesh Data Structures
- often want to store a set of triangles together which represent the tessellation of some polygon (2D) or surface (3D)
- simplest datastructure is just a list or (typically) array of independent triangles
- in 2D, we can store the coordinates for each triangle vertex using e.g. two floating point numbers
- total storage for independent tris:
data:image/s3,"s3://crabby-images/520b4/520b416dfe1cbe7865b20d1a38cd564e1ebc0f18" alt=""
- because meshes are usually used to store contiguous tessellations, i.e. where many triangles share edges and vertices, storing all three vertices explicitly for each triangle can be a waste of space
- also, storing independent triangles does not enforce any mesh connectivity: to move a shared vertex, the programmer must find all triangles which touch that vertex and move them each independently
- the next simplest datastructure is an array of indexed triangles where each triangle is specified by three integer indices into a separate vertex array
- total storage for indexed independent tris:
data:image/s3,"s3://crabby-images/c370a/c370a2a1622c3d5ec429110a2911db687af17124" alt=""
- this can save space when at least some triangles share common vertices
- also solves problem of editing a shared vertex
- can do better still
- recall the triangle fan that resulted from triangulation of a convex poly
- three vertices for first triangle
- only one vertex per triangle for all the rest
- vertices can either be given directly or indexed into a vertex array
- a triangle fan is thus an array of
vertices which defines a corresponding mesh of
triangles - each triangle
has vertices
in CCW order
data:image/s3,"s3://crabby-images/1817b/1817bf7f55c2943b67cad6e3216cdbb02998b91c" alt=""
- triangle fans are commonly used not only for convex polys but also for any star shaped poly
- a poly is star shaped if there exists some point inside the poly from which all other points in the poly can be “seen”
- total storage for indexed triangle fan:
data:image/s3,"s3://crabby-images/48702/487024af9ecee2f05f8f1152dccdba19d83f32fe" alt=""
- note that triangle fans only work when all triangles share a common vertex
- a more general structure which again only uses
vertices (or vertex indices) to represent
triangles is a triangle strip- basic idea is to generate triangles one by one, each time reusing two vertices of the previous triangle
- so each triangle
has vertices data:image/s3,"s3://crabby-images/9aae8/9aae8efef675294b8da95b8bce5a9a656d3aacbd" alt=""
- however, these will be in CCW order if
is even and CW order if
is odd
data:image/s3,"s3://crabby-images/e23aa/e23aa36dc6c555ba7ab10c9a30527e9d4e065614" alt=""
- common convention is to simply imply a reversal of the vertex order when reading/writing odd triangles from/to a strip
- total storage for triangle strip same as for triangle fan
- in practice, triangle meshes are often composed of multiple triangle arrays, each of which may be either independent triangles, a triangle fan, or a triangle strip. If indices are used, then a separate vertex array is also necessary.
Barycentric Coordinates
- recall the standard Cartesian (orthonormal) basis in 2D
, data:image/s3,"s3://crabby-images/f717b/f717b3e391ad93a3f2e231ded20ca2770b3cabe5" alt=""
- any point
in 2D can be specified as a 2-dimensional vector
of its coordinates - the interpretation is that the location of
is given by origin of the coordinate frame plus a vector formed by the linear combination of the basis vectors weighted by data:image/s3,"s3://crabby-images/60a26/60a26f8ba2c1975b5204abc4dbb6ae761106a8fe" alt=""
- if the origin is
then this works out to data:image/s3,"s3://crabby-images/34aa3/34aa379f7f53658d1553d3ea330763792768e494" alt=""
- this may seem tautological (true by construction), but that is because the standard basis at origin
is a special case: data:image/s3,"s3://crabby-images/6c747/6c7471dc04a3f45a515670eea07b4b7a913b02c6" alt=""
- however, we will need this level of formality to handle cases where the origin is not
and/or the basis is not the standard basis
- now we will develop a coordinate system based on arbitrary 2D triangle
- the extension to 3D is straightforward; we will study it later in the course
- let the triangle vertices be
in CCW order - we will consider a barycentric coordinate frame to be defined as a local frame relative to the Cartesian world frame in which the triangle is specified
- define the barycentric basis vectors as
and data:image/s3,"s3://crabby-images/14b67/14b6715553863e4cf69067ed9ef4e43adafc11ce" alt=""
- note that these may be non-orthonormal; i.e., they may not be perpendicular and/or they may not be unit length
- define the origin of the barycentric frame to be the first triangle vertex
data:image/s3,"s3://crabby-images/3703d/3703d07e6c9b9b03d84443ee0d8fb06cf73802b1" alt=""
- now any 2-vector
identifies a point relative to the barycentric basis and origin - we can compute the corresponding world frame point
using the same formula as above: data:image/s3,"s3://crabby-images/8b53d/8b53d7b97df667c277f47ef34fa19786fbe809cd" alt=""
- note that this works for any
, whether the point is inside or outside the triangle - thus, the barycentric basis and origin define a coordinate system for the entire 2D plane (as long as the triangle is not degenerate)
data:image/s3,"s3://crabby-images/8847c/8847c09513c3712669743f2f6aaf9b7e0596e8a0" alt=""
- one reason this is useful is that, with a little algebra, we can come up with simple conditions on
and
which hold if and only if
is inside the triangle - first, expand the above equation for
to use only the original triangle vertices:
are called the barycentric coordinates of data:image/s3,"s3://crabby-images/577fe/577fe60bb343eeee8cd3322568291358c640d38e" alt=""
- interpretation:
is a scaled distance from the edge
towards data:image/s3,"s3://crabby-images/3703d/3703d07e6c9b9b03d84443ee0d8fb06cf73802b1" alt=""
is a scaled distance from the edge
towards data:image/s3,"s3://crabby-images/04ef7/04ef73ef9c14af727cbc167a4618727ae69748ce" alt=""
is a scaled distance from the edge
towards data:image/s3,"s3://crabby-images/db2fd/db2fdb130e6fa0bae5c0aadf2b3355565d5b2c8b" alt=""
is inside the triangle (or on an edge) iff- or, equivalently,
is on a vertex when the above conditions hold and any two of
are both zero (this implies the remaining coordinate is 1)
is on an edge when the above conditions hold and any one of
is zero (this implies the remaining two coordinates sum to 1)- note that for all points on an edge, the two non-zero barycentric coordinates linearly interpolate from one vertex to the other
- in fact, barycentric coordinates also smoothly interpolate between all three vertices even in the interior of the triangle
- this is heavily used in graphics to smoothly interpolate coloring or shading information specified at vertices into the interior of a triangle
- recall that attributes such as color can be interpolated as well as coordinates
- especially in 3D, a common technique—called Gouraud shading—is to only calculate the effects of lighting in detail at the vertices, and then to interpolate through the interior
- this is not strictly correct, but it’s usually much faster than doing detailed calculations for every pixel, and it often looks fine as long as the triangles are relatively small
- we now know how to calculate the world-frame point
corresponding to barycentric coordinates
for a given triangle, but how to do the reverse? - want to find
given
(whether or not
is in the triangle) - recall that “plugging in” the coordinates of a point into the implicit equation for a line in 2D returns a value proportional to the signed perpendicular distance from the point to the line
- i.e. if the equation of a line is
then
is proportional to the signed perpendicular distance from
to the line - also recall that for any
is also an equation for the same line - idea: first find any implicit equation for the line through each triangle edge, then apply a scaling factor to each so that the opposite vertex is at value 1 of the corresponding barycentric coordinate
- from before, one way to produce an implicit equation for a line through two points
,
is data:image/s3,"s3://crabby-images/42de6/42de673dbbbc64e022198927062c54bc6c52e963" alt=""
- let
be the corresponding expression for barycentric coordinate data:image/s3,"s3://crabby-images/98801/9880130ba9006889077c31b7643bd93605d0c281" alt=""
be the corresponding expression for barycentric coordinate data:image/s3,"s3://crabby-images/3122d/3122d1ecfcbf510f8af5e5da1ccd38392d35ad68" alt=""
be the corresponding expression for barycentric coordinate data:image/s3,"s3://crabby-images/1c39f/1c39f7698450d038e96a3d535743ba6fe8fca29d" alt=""
- now just need to calculate appropriate scaling factors
so that - so these are just
- putting it all together, to find
given
, whether or not
is in the triangledata:image/s3,"s3://crabby-images/6b482/6b48211cddff3de0584752ce6a4d9a3b6f146289" alt=""
data:image/s3,"s3://crabby-images/0a7fc/0a7fcc0b905bb6c0700a2a75f2bbacc4aa5f16df" alt=""
data:image/s3,"s3://crabby-images/206b3/206b3f68e27154ce0d130820f7c02f0a04526134" alt=""
- where
and the original triangle coordinates are data:image/s3,"s3://crabby-images/8d5db/8d5db833a481edcaeb11d19177b585bbf6b1c30d" alt=""
Rasterizing Triangles
- now that we know, for a given triangle with CCW vertices
data:image/s3,"s3://crabby-images/8d5db/8d5db833a481edcaeb11d19177b585bbf6b1c30d" alt=""
- how to determine the barycentric coordinates
for any point
in world frame - how to determine if
are inside the triangle - how to interpolate vertex attributes, e.g. colors, using barycentric coordinates
- a reasonable triangle rasterization algorithm, i.e. an algorithm that takes
and turns on (or colors) the pixels inside the triangle, is simply
, data:image/s3,"s3://crabby-images/97c1f/97c1f3e9ff9e01f92c13a70dc0297604ddb50108" alt=""
, data:image/s3,"s3://crabby-images/c6a74/c6a74340e16cf16f863d1674f587772a7c37aed5" alt=""
- for
to data:image/s3,"s3://crabby-images/e5081/e508116017ea398d59f13ea620c74b4ea1c01def" alt=""
- for
to data:image/s3,"s3://crabby-images/2ba1a/2ba1aaf58755c4e4326672f7185b4b0e2d7b8e0d" alt=""
- compute barycentric coordinates
of data:image/s3,"s3://crabby-images/7292b/7292bda5aa6765e567fa2dfa6c4df3b69eaf5c82" alt=""
- if
and
and data:image/s3,"s3://crabby-images/ac704/ac704e2ae00f608d544546f04f9a005406038eeb" alt=""
- turn on pixel
data:image/s3,"s3://crabby-images/7292b/7292bda5aa6765e567fa2dfa6c4df3b69eaf5c82" alt=""
- or, if vertex colors
,
,
were given, set pixel
to color data:image/s3,"s3://crabby-images/52f46/52f46f5f9a962af1017ab7f9c784b3b97080529d" alt=""
- runtime scales as the size of the bounding rectangle of the triangle
- reasonably fast in practice, especially if care is taken to implement all calculations incrementally, in the same spirit as we did for rasterizing a line segment
- another practical issue to deal with is rasterizing two triangles that share an edge
- in practice, want each screen pixel to be “owned” by at most one triangle
- translucent triangle rendering looks bad otherwise
- “single ownership” can be be accomplished in a careful implementation
Next Time
- reading on website
- rigid and non-rigid transformations in 2D
- homogeneous coordinates
- scene graphs