614 lines
27 KiB
C++
614 lines
27 KiB
C++
/*
|
|
* Created by Brett Terpstra 6920201 on 22/10/22.
|
|
* Copyright (c) 2022 Brett Terpstra. All Rights Reserved.
|
|
*/
|
|
#include <graphics/graphics.h>
|
|
#include <chrono>
|
|
#include <graphics/gl/gl.h>
|
|
#include "engine/image/stb_image.h"
|
|
#include "graphics/debug_gui.h"
|
|
#include <engine/util/std.h>
|
|
|
|
namespace Raytracing {
|
|
|
|
extern Signals* RTSignal;
|
|
|
|
const std::vector<float> vertices = {
|
|
1.0f, 1.0f, 0.0f, // top right
|
|
1.0f, -1.0f, 0.0f, // bottom right
|
|
-1.0f, -1.0f, 0.0f, // bottom left
|
|
-1.0f, 1.0f, 0.0f // top left
|
|
};
|
|
|
|
const std::vector<unsigned int> indices = {
|
|
3, 1, 0, // first triangle
|
|
3, 2, 1 // second triangle
|
|
};
|
|
|
|
const std::vector<float> texCoords = {
|
|
1.0f, 1.0f, // top right
|
|
1.0f, 0.0f, // bottom right
|
|
0.0f, 0.0f, // bottom left
|
|
0.0f, 1.0f // top left
|
|
};
|
|
|
|
VAO* quad = nullptr;
|
|
|
|
void drawQuad() {
|
|
if (quad == nullptr)
|
|
quad = new VAO(vertices, texCoords, indices);
|
|
|
|
quad->bind();
|
|
quad->draw();
|
|
quad->unbind();
|
|
}
|
|
|
|
void deleteQuad() {
|
|
delete (quad);
|
|
}
|
|
|
|
struct ImageData {
|
|
unsigned char* data;
|
|
int width;
|
|
int height;
|
|
int channels;
|
|
};
|
|
|
|
GLFWimage getImageData(const std::string& file) {
|
|
GLFWimage imager;
|
|
int channels = 0;
|
|
imager.pixels = stbi_load(file.c_str(), &imager.width, &imager.height, &channels, 4);
|
|
return imager;
|
|
}
|
|
|
|
#ifdef USE_GLFW
|
|
XWindow::XWindow(int width, int height):
|
|
m_displayWidth(width), m_displayHeight(height) {
|
|
// OpenGL 4.6 is like 5 years old at this point and most systems support it
|
|
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 4);
|
|
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 6);
|
|
// We only generated GLAD headers for GL core profile. Plus renderdoc only support core profile.
|
|
glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
|
|
// Apple stuff. Not like apple supports GL4.6 anyways :/
|
|
glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);
|
|
glfwWindowHint(GLFW_VISIBLE, GL_TRUE);
|
|
glfwWindowHint(GLFW_DECORATED, GL_TRUE);
|
|
glfwWindowHint(GLFW_FOCUSED, GL_TRUE);
|
|
|
|
glfwSetErrorCallback([](int error_code, const char* description) -> void {
|
|
elog << "GLFW Error: " << error_code << "\n\t" << description << "\n";
|
|
});
|
|
|
|
if (!glfwInit())
|
|
throw std::runtime_error("Unable to init GLFW!\n");
|
|
|
|
// create a window using the above settings,
|
|
window = glfwCreateWindow(width, height, "GLFW 3P93 Raytracing Project", NULL, NULL);
|
|
if (!window)
|
|
throw std::runtime_error("Unable to create GLFW window!\n");
|
|
glfwMakeContextCurrent(window);
|
|
// enable V-Sync.
|
|
glfwSwapInterval(1);
|
|
// setup the callbacks we might use.
|
|
auto imageData16 = getImageData("../resources/icon/icon16.png");
|
|
auto imageData32 = getImageData("../resources/icon/icon32.png");
|
|
GLFWimage images[2];
|
|
// TODO: delete STBI_resize
|
|
images[0] = imageData16;
|
|
images[1] = imageData32;
|
|
|
|
glfwSetWindowIcon(window, 2, images);
|
|
stbi_image_free(imageData16.pixels);
|
|
stbi_image_free(imageData32.pixels);
|
|
|
|
|
|
glfwSetKeyCallback(window, [](GLFWwindow* _window, int key, int scancode, int action, int mods) -> void {
|
|
if (action == GLFW_PRESS)
|
|
Input::keyPressed(key);
|
|
else if (action == GLFW_RELEASE)
|
|
Input::keyReleased(key);
|
|
});
|
|
|
|
glfwSetMouseButtonCallback(window, [](GLFWwindow* _window, int button, int action, int mods) -> void {
|
|
if (action == GLFW_PRESS)
|
|
Input::mousePressed(button);
|
|
else if (action == GLFW_RELEASE)
|
|
Input::mouseReleased(button);
|
|
});
|
|
|
|
glfwSetCursorPosCallback(window, [](GLFWwindow* _window, double x, double y) -> void {
|
|
Input::moveMove(x, y);
|
|
});
|
|
|
|
IMGUI_CHECKVERSION();
|
|
ImGui::CreateContext();
|
|
ImGuiIO& io = ImGui::GetIO();
|
|
(void) io;
|
|
ImGui::StyleColorsDark();
|
|
|
|
// create ImGUI GLFW instance and install the callbacks
|
|
ImGui_ImplGlfw_InitForOpenGL(window, true);
|
|
ImGui_ImplOpenGL3_Init("#version 130");
|
|
|
|
int version = gladLoadGL(glfwGetProcAddress);
|
|
if (!version)
|
|
throw std::runtime_error("Unable to load Glad GL!\n");
|
|
|
|
glfwShowWindow(window);
|
|
ilog << "Loaded GL" << GLAD_VERSION_MAJOR(version) << "." << GLAD_VERSION_MINOR(version) << "!\n";
|
|
}
|
|
void XWindow::closeWindow() {
|
|
if (isCloseRequested)
|
|
return;
|
|
tlog << "Closing window!\n";
|
|
isCloseRequested = true;
|
|
ImGui_ImplOpenGL3_Shutdown();
|
|
ImGui_ImplGlfw_Shutdown();
|
|
ImGui::DestroyContext();
|
|
|
|
glfwDestroyWindow(window);
|
|
glfwTerminate();
|
|
}
|
|
void XWindow::beginUpdate() {
|
|
// check for window events
|
|
isCloseRequested = glfwWindowShouldClose(window);
|
|
// reset the current key-pressed state.
|
|
Input::state();
|
|
glfwPollEvents();
|
|
// update window settings
|
|
glfwGetFramebufferSize(window, &m_displayWidth, &m_displayHeight);
|
|
glViewport(0, 0, m_displayWidth, m_displayHeight);
|
|
glClearColor(0.5, 0.7, 1.0, 1.0);
|
|
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
|
glEnable(GL_DEPTH_TEST);
|
|
|
|
ImGui_ImplOpenGL3_NewFrame();
|
|
ImGui_ImplGlfw_NewFrame();
|
|
ImGui::NewFrame();
|
|
|
|
ImGui::ShowDemoWindow(nullptr);
|
|
}
|
|
void XWindow::endUpdate() {
|
|
// Render ImGUI
|
|
ImGui::Render();
|
|
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
|
|
|
|
glfwSwapBuffers(window);
|
|
auto _p1 = std::chrono::high_resolution_clock::now();
|
|
auto _now = std::chrono::duration_cast<std::chrono::nanoseconds>(_p1.time_since_epoch()).count();
|
|
long currentFrameTime = _now;
|
|
delta = double(currentFrameTime - lastFrameTime);
|
|
lastFrameTime = currentFrameTime;
|
|
frameTimeMs = delta / 1000000.0;
|
|
frameTimeS = delta / 1000000000.0;
|
|
fps = 1000 / frameTimeMs;
|
|
}
|
|
#else
|
|
XWindow::XWindow(int width, int height): m_width(width), m_height(height) {
|
|
// open the DEFAULT display. We don't want to open a specific screen as that is annoying.
|
|
dlog << "Creating X11 display!\n";
|
|
display = XOpenDisplay(NULL);
|
|
if (display == NULL)
|
|
throw std::runtime_error("Unable to open an X11 display! Is the X server running?");
|
|
// FBConfigs were added in GLX version 1.3.
|
|
if (!glXQueryVersion(display, &glx_major, &glx_minor))
|
|
throw std::runtime_error("Unable to get GLX version!");
|
|
if ((glx_major < 1) || (glx_major == 1 && glx_minor < 3))
|
|
throw std::runtime_error("Invalid GLX Version. At least 1.3 is required!");
|
|
// get the frame buffer config from the X11 window
|
|
frameBufferConfig = glXChooseFBConfig(display, DefaultScreen(display), visual_attribs, &frameBufferCount);
|
|
if (!frameBufferConfig)
|
|
throw std::runtime_error("Unable to get window framebuffer configs!");
|
|
dlog << "We have " << frameBufferCount << " framebuffers\n";
|
|
// select the best config from available ones.
|
|
int bestConfigIndex = 0, bestSamples = -1;
|
|
for (int i = 0; i < frameBufferCount; i++) {
|
|
XVisualInfo* xVisualInfo = glXGetVisualFromFBConfig(display, frameBufferConfig[i]);
|
|
if (xVisualInfo) {
|
|
int sampleBuffer, samples;
|
|
glXGetFBConfigAttrib(display, frameBufferConfig[i], GLX_SAMPLE_BUFFERS, &sampleBuffer);
|
|
glXGetFBConfigAttrib(display, frameBufferConfig[i], GLX_SAMPLES, &samples);
|
|
|
|
// if the sample buffer exists, and we have more samples in this config, make this config the one we use.
|
|
if (sampleBuffer && samples > bestSamples) {
|
|
bestConfigIndex = i;
|
|
bestSamples = samples;
|
|
}
|
|
}
|
|
XFree(xVisualInfo);
|
|
}
|
|
dlog << "We selected config: " << bestConfigIndex << " with " << bestSamples << "# of samples!\n";
|
|
GLXFBConfig bestConfig = frameBufferConfig[bestConfigIndex];
|
|
// we need to make sure we remember to free memory since we are working with c pointers!
|
|
XFree(frameBufferConfig);
|
|
// as I understand it every window in X11 is a sub-window of the root, or desktop window
|
|
// which is why I guess wayland was created, because X11 can't handle a bunch of stuff like VRF (variable refresh rate)
|
|
// because your say two monitors are treated as one big window, in effect limiting the refresh rate
|
|
// to whatever the lowest is. I still don't like Wayland though. Forced VSync and a fragmented design is annoying.
|
|
// plus needless security in a low level lib preventing stuff like discord screen sharing. Annoying as hell. /rant/.
|
|
desktop = DefaultRootWindow(display);
|
|
// try to open a gl visual context that meets our attributes' requirements
|
|
dlog << "Getting visual info!\n";
|
|
visualInfo = glXChooseVisual(display, 0, OpenGLAttributes);
|
|
// if our attributes are too much for the display, let's try reducing them. (modern hardware should support 24bit depth though)
|
|
if (visualInfo == NULL) {
|
|
wlog << "Unable to open a window with a depth of 24. Trying 16 bits.\n";
|
|
OpenGLAttributes[2] = 16;
|
|
visualInfo = glXChooseVisual(display, 0, OpenGLAttributes);
|
|
if (visualInfo == NULL) {
|
|
throw std::runtime_error("Unable to create window's visual context. Is your driver up to date?\n");
|
|
}
|
|
}
|
|
ilog << visualInfo->visualid << ": With depth: " << visualInfo->depth << " and RGB: " << visualInfo->bits_per_rgb << "\n";
|
|
// use our requirements to create a colormap for the screen.
|
|
colormap = XCreateColormap(display, desktop, visualInfo->visual, AllocNone);
|
|
// arguments used to open a window for us
|
|
xSetWindowAttributes.colormap = colormap;
|
|
// what kind of events we want to receive
|
|
xSetWindowAttributes.event_mask =
|
|
ExposureMask | KeyPressMask | ButtonPressMask | ButtonReleaseMask | ButtonMotionMask | KeyReleaseMask | LeaveWindowMask |
|
|
EnterWindowMask | FocusChangeMask | PointerMotionMask;
|
|
m_displayWidth = XDisplayWidth(display, 0);
|
|
m_displayHeight = XDisplayHeight(display, 0);
|
|
window = XCreateWindow(display,
|
|
desktop,
|
|
// center the display, even if the window manager ignores this.
|
|
m_displayWidth / 2 - width / 2,
|
|
m_displayHeight / 2 - height / 2,
|
|
width,
|
|
height,
|
|
0,
|
|
visualInfo->depth,
|
|
InputOutput,
|
|
visualInfo->visual,
|
|
CWColormap | CWEventMask,
|
|
&xSetWindowAttributes);
|
|
// install a error handler
|
|
// maybe we should set back the old one but I'd rather it goto std:err than crash the window
|
|
XSetErrorHandler([](Display* displayPtr, XErrorEvent* eventPtr) -> int {
|
|
elog << "An error occurred while trying to setup X11: " << eventPtr->error_code << ";\n " << eventPtr->minor_code << ";\n "
|
|
<< eventPtr->request_code << "\n";
|
|
return 0;
|
|
});
|
|
|
|
XStoreName(display, window, "3P93 Raytracing Project");
|
|
ImageInput image("../resources/icon/icon.png");
|
|
auto* imageData = image.getImageAsIconBuffer();
|
|
|
|
auto hints = XAllocWMHints();
|
|
hints->flags = IconPixmapHint | StateHint | IconPositionHint;
|
|
auto pixMapRob = XCreateBitmapFromData(display, window, (const char*) (icon_bits), 32, 32);
|
|
if (!pixMapRob)
|
|
flog << "Unable to create icon pixel map\n";
|
|
hints->icon_pixmap = pixMapRob;
|
|
hints->initial_state = IconicState;
|
|
hints->icon_x = 0;
|
|
hints->icon_y = 0;
|
|
XSetWMHints(display, window, hints);
|
|
XFree(hints);
|
|
|
|
int length = 32 * 32 + 2;
|
|
//int length = 16 * 16 * 4 + 32*32 * 4 + 4 * 4;
|
|
XChangeProperty(display,
|
|
window,
|
|
XInternAtom(display, "_NET_WM_ICON", False),
|
|
XInternAtom(display, "CARDINAL", False),
|
|
32,
|
|
PropModeReplace,
|
|
(const unsigned char*) imageData,
|
|
length);
|
|
delete[](imageData);
|
|
// there might actually be an argument to be made about X11 being outdated....
|
|
wmDelete = XInternAtom(display, "WM_DELETE_WINDOW", True);
|
|
XSetWMProtocols(display, window, &wmDelete, 1);
|
|
|
|
// Now show the window
|
|
XMapWindow(display, window);
|
|
|
|
// get the list of GLX extensions for this system
|
|
const char* glExtensions = glXQueryExtensionsString(display, DefaultScreen(display));
|
|
// much in the same way that we get GL function pointers and use them we will do the same with the context creation
|
|
auto glXCreateContextAttribsARBPtr = (PFNGLXCREATECONTEXTATTRIBSARBPROC) glXGetProcAddressARB((unsigned char*) "glXCreateContextAttribsARB");
|
|
// now we can finally create a OpenGL context for our window
|
|
int OpenGLContextAttributes[] = {
|
|
// OpenGL major version, we want GL4.5+
|
|
GLX_CONTEXT_MAJOR_VERSION_ARB, 4,
|
|
// OpenGL minor version,
|
|
GLX_CONTEXT_MINOR_VERSION_ARB, 5,
|
|
// I don't remember what this does, but I know GLFW recommends that forward compatability be set true, (Pretty sure it's only an issue
|
|
// on MacOS but I've always included this in all my projects so :shrug:
|
|
GLX_CONTEXT_FLAGS_ARB, GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB,
|
|
// Core profile for better Renderdoc compatibility + I don't need non core extensions
|
|
GLX_CONTEXT_PROFILE_MASK_ARB, GLX_CONTEXT_CORE_PROFILE_BIT_ARB,
|
|
None
|
|
};
|
|
// now we can actually create and acquire the context
|
|
glContext = glXCreateContextAttribsARBPtr(display, bestConfig, 0, True, OpenGLContextAttributes);
|
|
// make sure if there was any error we are notified
|
|
XSync(display, False);
|
|
if (!glContext)
|
|
flog << "Unable to create GL context!";
|
|
if (glXIsDirect(display, glContext)) {
|
|
ilog << "A direct GL context was acquired!\n";
|
|
} else // direct contexts are faster than indirect!
|
|
wlog << "Warning! Indirect context!\n";
|
|
// make the currently executing thread the one current to the OpenGL context
|
|
// since OpenGL is a single threaded finite state machine if we want to do mutli-threading with OpenGL (we don't)
|
|
// this has to be called in each thread before we make use of any OpenGL function.
|
|
glXMakeCurrent(display, window, glContext);
|
|
// Now we can issue some OpenGL commands
|
|
// we want to respect depth
|
|
glEnable(GL_DEPTH_TEST);
|
|
|
|
//assignGLFunctionPointers();
|
|
//glEnableVertexArrayAttribPtr = glXGetProcAddress((unsigned char*)("glEnableVertexArrayAttrib"));
|
|
|
|
// Setup Dear IMGUI
|
|
IMGUI_CHECKVERSION();
|
|
ImGui::CreateContext();
|
|
ImGuiIO& io = ImGui::GetIO();
|
|
|
|
ImGui::StyleColorsDark();
|
|
|
|
ImGui_ImplGlfw_Init();
|
|
ImGui_ImplOpenGL3_Init("#version 130");
|
|
|
|
|
|
}
|
|
void XWindow::beginUpdate() {
|
|
// only try to check events if they are queued
|
|
while (XPending(display) > 0) {
|
|
XNextEvent(display, &events);
|
|
|
|
// one of the few times I'll use a switch statement
|
|
switch (events.type) {
|
|
// called when the system thinks that the window should be updated
|
|
// so on window resize events and of course on actual window update
|
|
case Expose:
|
|
// update window information
|
|
XGetWindowAttributes(display, window, &windowAttributes);
|
|
glViewport(0, 0, windowAttributes.width, windowAttributes.height);
|
|
this->m_width = windowAttributes.width;
|
|
this->m_height = windowAttributes.height;
|
|
break;
|
|
case KeyPress: {
|
|
// translates xkeycodes to ascii keys
|
|
KeySym _key = XLookupKeysym(&events.xkey, 0);
|
|
ImGui_ImplGlfw_KeyCallback(_key, 0, true, 0);
|
|
break;
|
|
}
|
|
case KeyRelease: {
|
|
KeySym _key = XLookupKeysym(&events.xkey, 0);
|
|
ImGui_ImplGlfw_KeyCallback(_key, 0, false, 0);
|
|
break;
|
|
}
|
|
case ButtonPress:
|
|
if (events.xbutton.button < 4)
|
|
ImGui_ImplGlfw_MouseButtonCallback(events.xbutton.button - 1, true, 0);
|
|
else {
|
|
if (events.xbutton.button == 4)
|
|
ImGui_ImplGlfw_ScrollCallback(0, 1);
|
|
else if (events.xbutton.button == 5)
|
|
ImGui_ImplGlfw_ScrollCallback(0, -1);
|
|
else if (events.xbutton.button == 6)
|
|
ImGui_ImplGlfw_ScrollCallback(1, 0);
|
|
else if (events.xbutton.button == 7)
|
|
ImGui_ImplGlfw_ScrollCallback(-1, 0);
|
|
}
|
|
break;
|
|
case ButtonRelease:
|
|
if (events.xbutton.button < 4)
|
|
ImGui_ImplGlfw_MouseButtonCallback(events.xbutton.button - 1, false, 0);
|
|
else {
|
|
ImGui_ImplGlfw_ScrollCallback(0, 0);
|
|
}
|
|
break;
|
|
case MotionNotify:
|
|
ImGui_ImplGlfw_CursorPosCallback(events.xmotion.x, events.xmotion.y, true);
|
|
break;
|
|
case EnterNotify:
|
|
ImGui_ImplGlfw_CursorEnterCallback(1, true);
|
|
break;
|
|
case LeaveNotify:
|
|
ImGui_ImplGlfw_CursorEnterCallback(0, true);
|
|
break;
|
|
case FocusIn:
|
|
ImGui_ImplGlfw_WindowFocusCallback(1);
|
|
break;
|
|
case FocusOut:
|
|
ImGui_ImplGlfw_WindowFocusCallback(0);
|
|
break;
|
|
case ClientMessage:
|
|
if (events.xclient.data.l[0] == wmDelete) {
|
|
closeWindow();
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
glClearColor(0, 0, 0, 1);
|
|
glClear(GL_DEPTH_BUFFER_BIT | GL_COLOR_BUFFER_BIT);
|
|
ImGui_ImplOpenGL3_NewFrame();
|
|
ImGui_ImplGlfw_NewFrame((double) std::chrono::duration_cast<std::chrono::milliseconds>(
|
|
std::chrono::high_resolution_clock::now().time_since_epoch()).count(),
|
|
m_width,
|
|
m_height,
|
|
m_width,
|
|
m_height);
|
|
ImGui::NewFrame();
|
|
|
|
static bool t = true;
|
|
if (t)
|
|
ImGui::ShowDemoWindow(&t);
|
|
}
|
|
void XWindow::endUpdate() {
|
|
ImGui::Render();
|
|
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
|
|
ImGui_ImplGlfw_UpdateMouseData(true, true);
|
|
|
|
// we use double buffering to prevent screen tearing and other visual disturbances
|
|
glXSwapBuffers(display, window);
|
|
}
|
|
void XWindow::closeWindow() {
|
|
// since this is called in the destructor, we don't want to double delete our window
|
|
if (isCloseRequested)
|
|
return;
|
|
tlog << "Closing window!\n";
|
|
isCloseRequested = true;
|
|
ImGui_ImplOpenGL3_Shutdown();
|
|
ImGui_ImplGlfw_Shutdown();
|
|
ImGui::DestroyContext();
|
|
|
|
XFree(visualInfo);
|
|
glXMakeCurrent(display, None, NULL);
|
|
glXDestroyContext(display, glContext);
|
|
XDestroyWindow(display, window);
|
|
XCloseDisplay(display);
|
|
}
|
|
#endif
|
|
XWindow::~XWindow() {
|
|
closeWindow();
|
|
deleteKeys();
|
|
}
|
|
void XWindow::setMouseGrabbed(bool grabbed) {
|
|
glfwSetInputMode(window, GLFW_CURSOR, grabbed ? GLFW_CURSOR_DISABLED : GLFW_CURSOR_NORMAL);
|
|
}
|
|
bool XWindow::isMouseGrabbed() {
|
|
return glfwGetInputMode(window, GLFW_CURSOR) == GLFW_CURSOR_DISABLED;
|
|
}
|
|
|
|
static bool started = false, debug = false;
|
|
static int maxRayBounce = 50;
|
|
static int raysPerPixel = 50;
|
|
static float yaw = 0, pitch = 0;
|
|
|
|
std::pair<Mat4x4, Mat4x4> DisplayRenderer::getCameraMatrices() {
|
|
auto projection = m_camera.project();
|
|
if (m_window.isMouseGrabbed()) {
|
|
yaw += (float) Input::getMouseDelta().x * (1000.0f / ImGui::GetIO().Framerate / 1000.0f) * 3;
|
|
pitch += (float) Input::getMouseDelta().y * (1000.0f / ImGui::GetIO().Framerate / 1000.0f) * 1.5f;
|
|
|
|
const PRECISION_TYPE turnSpeed = 50;
|
|
|
|
if (Input::isKeyDown(GLFW_KEY_LEFT))
|
|
yaw += float(turnSpeed * m_window.getFrameTimeSeconds());
|
|
if (Input::isKeyDown(GLFW_KEY_RIGHT))
|
|
yaw -= float(turnSpeed * m_window.getFrameTimeSeconds());
|
|
if (Input::isKeyDown(GLFW_KEY_UP))
|
|
pitch += float(turnSpeed * m_window.getFrameTimeSeconds());
|
|
if (Input::isKeyDown(GLFW_KEY_DOWN))
|
|
pitch -= float(turnSpeed * m_window.getFrameTimeSeconds());
|
|
|
|
PRECISION_TYPE moveAtX = 0, moveAtY = 0, moveAtZ = 0;
|
|
PRECISION_TYPE speed = 40.0f;
|
|
|
|
if (Input::isKeyDown(GLFW_KEY_LEFT_ALT))
|
|
speed = 5.0f;
|
|
if (Input::isKeyDown(GLFW_KEY_LEFT_CONTROL))
|
|
speed = speed * 2;
|
|
|
|
if (Input::isKeyDown(GLFW_KEY_W))
|
|
moveAtX = speed * m_window.getFrameTimeSeconds();
|
|
else if (Input::isKeyDown(GLFW_KEY_S))
|
|
moveAtX = -speed * m_window.getFrameTimeSeconds();
|
|
|
|
if (Input::isKeyDown(GLFW_KEY_A))
|
|
moveAtZ = -speed * m_window.getFrameTimeSeconds();
|
|
else if (Input::isKeyDown(GLFW_KEY_D))
|
|
moveAtZ = speed * m_window.getFrameTimeSeconds();
|
|
|
|
if (Input::isKeyDown(GLFW_KEY_SPACE))
|
|
moveAtY = speed * m_window.getFrameTimeSeconds();
|
|
else if (Input::isKeyDown(GLFW_KEY_LEFT_SHIFT))
|
|
moveAtY = -speed * m_window.getFrameTimeSeconds();
|
|
|
|
PRECISION_TYPE radYaw = degreeeToRadian(yaw);
|
|
|
|
PRECISION_TYPE deltaX = -moveAtX * std::sin(radYaw) + moveAtZ * std::cos(radYaw);
|
|
PRECISION_TYPE deltaY = moveAtY;
|
|
PRECISION_TYPE deltaZ = -moveAtX * std::cos(radYaw) + -moveAtZ * std::sin(radYaw);
|
|
|
|
m_camera.setPosition(m_camera.getPosition() + Vec4{deltaX, deltaY, deltaZ});
|
|
}
|
|
auto view = m_camera.view(yaw, pitch);
|
|
return {projection, view};
|
|
}
|
|
|
|
void DisplayRenderer::draw() {
|
|
if (RTSignal->haltExecution) { m_window.closeWindow(); }
|
|
if (Input::isKeyDown(GLFW_KEY_ESCAPE) && Input::isState(GLFW_KEY_ESCAPE))
|
|
m_window.setMouseGrabbed(!m_window.isMouseGrabbed());
|
|
|
|
DebugUI::render([this]() -> void {
|
|
if (ImGui::Button("Start") && !started) {
|
|
started = true;
|
|
RTSignal->haltRaytracing = false;
|
|
ilog << "Running raycaster!\n";
|
|
// we don't actually have to check for --single since it's implied to be default true.
|
|
int threads = std::stoi(m_parser.getOptionValue("--threads"));
|
|
if (m_parser.hasOption("--mpi")) {
|
|
//m_raycaster.runMPI(raycaster.partitionScreen());
|
|
} else if (m_parser.hasOption("--openmp")) {
|
|
m_raycaster.runOpenMP(threads);
|
|
} else {
|
|
m_raycaster.runSTDThread(threads);
|
|
}
|
|
}
|
|
if (ImGui::Checkbox("Pause", &RTSignal->pauseRaytracing)) {}
|
|
if (ImGui::Button("Stop") && started) {
|
|
RTSignal->haltRaytracing = true;
|
|
started = false;
|
|
m_raycaster.deleteThreads();
|
|
}
|
|
ImGui::NewLine();
|
|
ImGui::InputInt("Max Ray Bounce", &maxRayBounce);
|
|
ImGui::InputInt("Rays Per Pixel", &raysPerPixel);
|
|
m_raycaster.updateRayInfo(maxRayBounce, raysPerPixel);
|
|
ImGui::Checkbox("Debug", &debug);
|
|
});
|
|
|
|
// we want to be able to move around, and the camera matrix functions automatically recalculate image region & projection data.
|
|
if (m_parser.hasOption("--gpu")) {
|
|
getCameraMatrices();
|
|
}
|
|
|
|
if (debug) {
|
|
// if (Input::isKeyDown(GLFW_KEY_E) && Input::isState(GLFW_KEY_E)) {
|
|
// auto ray = m_camera.projectRay((PRECISION_TYPE) m_window.displayWidth() / 2, (PRECISION_TYPE) m_window.displayHeight() / 2);
|
|
//
|
|
// //auto results = m_world.checkIfHit(ray, 0, 1000).first;
|
|
// auto bvh = m_world.getBVH()->rayAnyHitIntersect(ray, 0, 1000);
|
|
// //if (results.hit)
|
|
// // ilog << "World Results: " << results.hitPoint << " " << results.length << "\n";
|
|
// //else
|
|
// // ilog << "World not hit.\n";
|
|
// if (!bvh.empty())
|
|
// ilog << "BVH Results: " << bvh.size() << " " << bvh[0].ptr->getPosition() << "\n";
|
|
// else
|
|
// ilog << "BVH not hit.\n";
|
|
// }
|
|
// if (Input::isKeyDown(GLFW_KEY_R) && Input::isState(GLFW_KEY_R))
|
|
// m_world.getBVH()->resetNodes();
|
|
auto matrices = getCameraMatrices();
|
|
m_worldShader.setMatrix("projectMatrix", matrices.first);
|
|
m_worldShader.setMatrix("viewMatrix", matrices.second);
|
|
m_worldShader.use();
|
|
auto objs = m_world.getObjectsInWorld();
|
|
for (auto obj: objs) {
|
|
if (obj->getVAO() != nullptr) {
|
|
obj->getVAO()->bind();
|
|
obj->getVAO()->draw(m_worldShader, {obj->getPosition()});
|
|
}
|
|
}
|
|
DebugMenus::render();
|
|
} else {
|
|
m_imageShader.use();
|
|
m_mainImage.updateImage();
|
|
m_mainImage.bind();
|
|
m_mainImage.enableGlTextures(1);
|
|
drawQuad();
|
|
}
|
|
|
|
m_mainImage.updateImage();
|
|
}
|
|
} |