COSC-3P93-Project/Step 2/src/raytracing.cpp

70 lines
2.9 KiB
C++
Raw Normal View History

2022-10-16 17:53:33 -04:00
/*
* Created by Brett Terpstra 6920201 on 16/10/22.
* Copyright (c) 2022 Brett Terpstra. All Rights Reserved.
*/
#include <raytracing.h>
2022-10-17 00:29:34 -04:00
#include <queue>
2022-10-16 17:53:33 -04:00
namespace Raytracing {
Ray Camera::projectRay(PRECISION_TYPE x, PRECISION_TYPE y) {
// transform the x and y to points from image coords to be inside the camera's viewport.
double transformedX = (x / (image.getWidth() - 1));
auto transformedY = (y / (image.getHeight() - 1));
// then generate a ray which extends out from the camera position in the direction with respects to its position on the image
return {position, imageOrigin + transformedX * horizontalAxis + transformedY * verticalAxis - position};
}
void Camera::lookAt(const vec4& pos, const vec4& lookAtPos, const vec4& up) {
// standard camera lookAt function
auto w = (pos - lookAtPos).normalize();
auto u = (vec4::cross(up, w)).normalize();
auto v = vec4::cross(w, u);
position = pos;
horizontalAxis = viewportWidth * u;
verticalAxis = viewportHeight * v;
imageOrigin = position - horizontalAxis/2 - verticalAxis/2 - w;
}
void Camera::setRotation(const PRECISION_TYPE yaw, const PRECISION_TYPE pitch, const PRECISION_TYPE roll) {
// TODO:
}
2022-10-17 00:29:34 -04:00
void Raycaster::run() {
for (int i = 0; i < image.getWidth(); i++){
for (int j = 0; j < image.getHeight(); j++){
Raytracing::vec4 color;
// TODO: profile for speed;
for (int s = 0; s < raysPerPixel; s++){
// simulate anti aliasing by generating rays with very slight random directions
// TODO: this should be updated with a [-1, 1] generator as this currently only produces [0, 1]
// and therefore lacks true aliasing
color = color + raycast(camera.projectRay(i + rnd.getDouble(), j + rnd.getDouble()), 0);
}
PRECISION_TYPE sf = 1.0 / raysPerPixel;
// apply pixel color with gamma correction
image.setPixelColor(i, j, {std::sqrt(sf * color.r()), std::sqrt(sf * color.g()), std::sqrt(sf * color.b())});
}
}
}
2022-10-16 17:53:33 -04:00
2022-10-17 00:29:34 -04:00
vec4 Raycaster::raycast(const Ray& ray, int depth) {
if (depth > maxBounceDepth)
return {0,0,0};
2022-10-16 17:53:33 -04:00
2022-10-17 00:29:34 -04:00
auto hit = world.checkIfHit(ray, 0.001, infinity);
if (hit.hit) {
// randomly cast our bounce rays to simulate diffuse lighting
vec4 newRay = hit.hitPoint + hit.normal + randomInSphere();
// recursion is the only good way to do this
// TODO: maybe without the recursion, clang tidy is annoying me.
return 0.5* raycast({hit.hitPoint, newRay - hit.hitPoint}, depth + 1);
2022-10-16 17:53:33 -04:00
}
2022-10-16 19:24:37 -04:00
2022-10-17 00:29:34 -04:00
vec4 dir = ray.getDirection().normalize();
auto t = 0.5f * (dir.y() + 1.0);
return (1.0 - t) * vec4(1.0, 1.0, 1.0) + t * vec4(0.5, 0.7, 1.0);
2022-10-16 17:53:33 -04:00
}
}