| CSCI 441 - Computer GraphicsFall 2021 - Side Quest 2 - RaytracingCanvas | CS @ Mines | Mines | Piazza | |||
| | Home | Assignments | Leaderboard | Schedule | Resources | | |||
|  2015: Asgard  2016: Mount Olympus  2017: Aaru Park  2018: Findias Speedway  2019: Hanan Pacha  2020: Onogoro Island  2021: The Colosseum | |||
| This side quest is due by December 14, 2021 by 11:59pm to earn the achievement. Part I - Raytracing
    For this Side Quest, you will write a basic raytracer and implement some of the functionality that raytracers
    enable you as a programmer to add with relative ease. You will start the assignment with base code, and will
    implement a number of elements of functionality:
     
   a rendered frame from the raytracer.   the previous frame, in OpenGL preview mode. Base Code
    You can download the base code this assignment here. There are a number of classes that encapsulate the work
    that needs to be done in an OpenGL pipeline; familiar to you should be the  ReflectiveMaterial,
    andPointLightclasses, which are all very similar to the classes you have used or created for previous or upcoming homework assignments.
    There are a few new classes that are also particularly relevant to this assignment. They are:Ray:
    this class is a storage class for rays: it holds two values, representing an origin and a direction.Shape:
    this is an abstract class that allows us to lump all of our renderable objects into the same vector inside of main.cpp.
    Instead of having separate arrays forSpheres,Planes,Triangles,Meshes, etc., all of those classes
    extend theShapeclass and can be used interchangeably. (i.e. regardless of which object type it is callshapes[i]->drawViaOpenGL();) All
    classes that inherit from theShapeclass must implement thedrawViaOpenGL()function, which draws the object using OpenGL calls,
    and theintersectWithRay()function, which computes the intersection between a ray and the object and returns information about
    the intersection, if it occurred.Sphere:
    has a position (center of the sphere) and a radius. Already contains an implementeddrawViaOpenGL()function so that it
    can be used in Preview Mode. You will need to implement itsintersectWithRay()function.Plane:
    defined by a point on the plane and the normal of the plane. Already contains an implementeddrawViaOpenGL()function, but
    note that it draws the plane as a finite, bounded quad (in reality, the plane extends infinitely). You will need to implement itsintersectWithRay()function.IntersectionData:
    this is a storage class to hold information about an intersection, if one occurs. It is returned from theintersectWithRay()functions: if the ray passed into that function intersected that object, then thewasValidIntersectionboolean member variable should be set.
    If this intersection was valid, then its member variables will hold a) the point of intersection, b) the surface normal of the object at the point of intersection,
    and c) a copy of the shape'sMaterialobject. Your intersection functions must fill these member variables.Places where you need to write code are indicated with TODOs. Pressing 'r' calls a raytraceImage()function which is
    located in Raytracer.cpp. This function needs to fill each pixel with a color based on the scene.Camera Rays
    The first step of your raytracer is computing the camera ray (primary ray) for each pixel of an image.
    For more information on computing primary rays, see this website.
    Note that thanks to our OpenGL formulation, we have the vertical field of view and must compute the horizontal field of view. You may find it easier to think of the terms involved in camera ray computation as a vector math problem. If we assume that the image plane is a unit distance away from the camera, we can get a point on the center of the image plane by adding the camera's position to its direction. We can compute the width and height of the image plane using the tangent functions as described in the link above. Lastly, we can determine the vectors that lie in the image plane -- the first is the camera's local X axis, perpendicular to the viewing direction and global up vector (0,1,0), and the second is the camera's local Y axis, perpendicular to its local X axis and local Z axis (direction) -- and move along these vectors based on the percentage of the image plane's width and height. Intersection Testing and Shading
    After being able to calculate the camera ray through each pixel, you need to intersect that ray against all objects in the scene and
    save the nearest intersection, if any. This can be done by iterating over each shape and calling its intersectWithRay() function -- but
    you will need to implement these functions for the  SphereandPlaneclasses. Start withPlaneand move on toSphereafter that is confirmed to work.Once you know a ray has intersected with an object, and your intersectWithRay()function has appropriately filled in the intersection point,
    surface normal, and material in theIntersectionDataobject, you must compute a final color for the pixel. You must do this using
    the Phong Reflectance Model as discussed in class. Basically, shading a pixel requires iterating over all lights and
    summing the contribution from each light.Shadows and Reflections (& Refractions)
    Before computing the diffuse and specular contributions from a light, you must determine whether that point is in shadow. To do this, you need
    to test a ray for intersection against the scene whose origin is the intersection point and whose direction points toward that light. (NOTE:
    to prevent rounding errors, move the intersection point a small amount along the direction vector before doing testing.) If there are any intersections,
    the point should receive only the ambient component of the lighting equation. Lastly, you will notice that the Material class has one extra parameter -- reflectivity. When shading a point, if the reflectivity of
    an object is greater than zero, you must cast a reflection ray to determine the reflected color. NOTE: as with
    shadow rays applies: you should move the intersection point a small amount along the reflection vector to avoid rounding errors. After finding
    the reflected color, you lerp the final color of the point asreflectivity * reflectedColor + (1 - reflectivity)*phongColor.  reflective spheres (note: this scene has two lights) Additionally, the Material class supports transparency and refraction. Only reflection is required but you are encouraged to add in a transparent & refractive implementation as well.   a refractive sphere (note: the sphere and shadow reflect the transparency) Structuring Your Code
    Since the operations involved with determining shadows and reflection are recursive, it makes sense to write functions to support this. For instance,
    use the function called  intersectRayAgainstScene()that intersects a ray against all objects and returns the closest intersection, if any.
    This will keep yourraytraceImage()function much cleaner. Additionally, you have a second function,shadeIntersection(),
    that computes the color for a given intersection point and returns that. It makes sense to put this as its own function, since you will have to
    handle shadow rays (use theintersectRayAgainstScene()function) and reflection rays (intersectRayAgainstScene()with a reflection
    ray and then recursively callshadeIntersection()to determine the reflected color). For shadow rays, you may want to write a version ofintersectRayAgainstScene()that returns true as soon as possible if it intersects any object, and doesn't bother computing the nearest
    intersection, for speed purposes. Additionally, you may want to clamp your color value before returning it, in case
    it lies outside the range of [0,1].
 | |||
|   
                    Web +50 XP
                 |   
                    Dr. Ray Stantz
                 | 
HeroName_SQ2.zip.
        Upload this file to Canvas under SQ2.
 
     
    