Vector3.h
double SquaredLength() const
{
return x * x + y * y + z * z;
}
static Vector3 Random()
{
return Vector3(RandomDouble(), RandomDouble(), RandomDouble());
}
static Vector3 Random(double min, double max)
{
return Vector3(RandomDouble(min, max), RandomDouble(min, max), RandomDouble(min, max));
}
};
...
inline Vector3 Unit(const Vector3& vector)
{
return vector / vector.Length();
}
inline Position RandomInUnitSphere()
{
while(true)
{
Position position = Vector3::Random(-1, 1);
if(position.SquaredLength() < 1) return position;
}
}
inline Vector3 RandomUnitVector()
{
return Unit(RandomInUnitSphere());
}
inline Vector3 RandomOnHemisphere(const Vector3& normal)
{
Vector3 onUnitSphere = RandomUnitVector();
// If in the same hemisphere as the normal
if(Dot(onUnitSphere, normal) > 0.0)
{
return onUnitSphere;
}
return -onUnitSphere;
}
Camera.cpp
Color Camera::RayColor(const Ray& rRay, const Hittable& rWorld) const
{
HitInfo hitInfo;
if (rWorld.Hit(rRay, Interval(0, infinity), hitInfo)) {
Vector3 direction = RandomOnHemisphere(hitInfo.normal);
return 0.5 * RayColor(Ray(hitInfo.coordinates, direction), rWorld); // Recursion
}
Vector3 unitDirection = Unit(rRay.GetDirection());
double blue = 0.5 * (unitDirection.y + 1.0);
return (1.0 - blue) * Color(1.0, 1.0, 1.0) + blue * Color (0.5, 0.7, 1.0);
}
Camera.h
class Camera
{
public:
Camera() = default;
Camera(double imageWidth, double ratio, int samplePerPixel = 10, int bounces = 10): // here
aspectRatio(ratio), width(imageWidth), sampleCount(samplePerPixel), maxBounces(bounces){} // here too
void Render(const Hittable& rWorld);
private:
int height;
double aspectRatio, width;
int sampleCount;
int maxBounces; // Here
Position center, originPixelLocation;
Vector3 pixelDeltaX, pixelDeltaY;
void Initialize();
Color RayColor(const Ray& rRay, int bouncesLeft, const Hittable& rWorld) const; // and here
Ray GetRay(int x, int y) const;
Vector3 PixelSampleSquared() const;
};
Camera.cpp
void Camera::Render(const Hittable& rWorld)
{
...
for(int sample = 0; sample < sampleCount; sample ++)
{
Ray ray = GetRay(x, y);
pixel += RayColor(ray, maxBounces, rWorld);
}
...
}
...
Color Camera::RayColor(const Ray& rRay, int bouncesLeft, const Hittable& rWorld) const
{
HitInfo hitInfo;
if(bouncesLeft <= 0) return Color(0, 0, 0);
if (rWorld.Hit(rRay, Interval(0, infinity), hitInfo)) {
Vector3 direction = RandomOnHemisphere(hitInfo.normal);
return 0.5 * RayColor(Ray(hitInfo.coordinates, direction), bouncesLeft-1, rWorld);
}
...
}
Raytracing.cpp
#include "Camera.h"
#include "HittableCollection.h"
#include "Sphere.h"
using namespace std;
int main(int argc, char* argv[])
{
// World
HittableCollection world;
world.Add(make_shared<Sphere>(Position(0, 0, -1), 0.5));
world.Add(make_shared<Sphere>(Position(0, -100.5, -1), 100));
Camera camera(400, 16.0/9.0, 100, 50);
camera.Render(world);
return 0;
}
Camera.cpp
Color Camera::RayColor(const Ray& rRay, int bouncesLeft, const Hittable& rWorld) const
{
HitInfo hitInfo;
if(bouncesLeft <= 0) return Color(0, 0, 0);
if (rWorld.Hit(rRay, Interval(0.001, infinity), hitInfo)) { // Hi
Vector3 direction = RandomOnHemisphere(hitInfo.normal);
return 0.5 * RayColor(Ray(hitInfo.coordinates, direction), bouncesLeft-1, rWorld);
}
Vector3 unitDirection = Unit(rRay.GetDirection());
double blue = 0.5 * (unitDirection.y + 1.0);
return (1.0 - blue) * Color(1.0, 1.0, 1.0) + blue * Color (0.5, 0.7, 1.0);
}
Camera.cpp
Color Camera::RayColor(const Ray& rRay, int bouncesLeft, const Hittable& rWorld) const
{
HitInfo hitInfo;
if(bouncesLeft <= 0) return Color(0, 0, 0);
if (rWorld.Hit(rRay, Interval(0.001, infinity), hitInfo)) {
Vector3 direction = hitInfo.normal + RandomUnitVector();
return 0.5 * RayColor(Ray(hitInfo.coordinates, direction), bouncesLeft-1, rWorld);
}
Vector3 unitDirection = Unit(rRay.GetDirection());
double blue = 0.5 * (unitDirection.y + 1.0);
return (1.0 - blue) * Color(1.0, 1.0, 1.0) + blue * Color (0.5, 0.7, 1.0);
}
Camera.cpp
Color Camera::RayColor(const Ray& rRay, int bouncesLeft, const Hittable& rWorld) const
{
HitInfo hitInfo;
if(bouncesLeft <= 0) return Color(0, 0, 0);
if (rWorld.Hit(rRay, Interval(0.001, infinity), hitInfo)) {
Vector3 direction = hitInfo.normal + RandomUnitVector();
return 0.9 * RayColor(Ray(hitInfo.coordinates, direction), bouncesLeft-1, rWorld); // The "0.9" is the Reflectance Factor
}
Vector3 unitDirection = Unit(rRay.GetDirection());
double blue = 0.5 * (unitDirection.y + 1.0);
return (1.0 - blue) * Color(1.0, 1.0, 1.0) + blue * Color (0.5, 0.7, 1.0);
}
Color.h
inline double LinearToGamma(double linearComponent)
{
return sqrt(linearComponent);
}
inline void WriteColor(std::ostream &out, Color pixel, int sampleCount)
{
double scale = 1.0 / sampleCount;
double r = LinearToGamma(pixel.x * scale);
double g = LinearToGamma(pixel.y * scale);
double b = LinearToGamma(pixel.z * scale);
// Write the translated [0,255] value of each color component.
static const Interval intensity(0.000, 0.999);
out << static_cast<int>(255.999 * intensity.Clamp(r)) << ' '
<< static_cast<int>(255.999 * intensity.Clamp(g)) << ' '
<< static_cast<int>(255.999 * intensity.Clamp(b)) << '\n';
}