SphericalHarmonics/src/main.cpp

252 lines
6.7 KiB
C++
Raw Normal View History

2021-11-16 21:57:55 +00:00
#include <iostream>
#include <chrono>
#include <glad/glad.h>
#include <GLFW/glfw3.h>
#include <glm/gtc/matrix_transform.hpp>
2021-11-18 14:41:56 +00:00
#include <backends/imgui_impl_glfw.h>
#include <backends/imgui_impl_opengl3.h>
2021-11-16 21:57:55 +00:00
#include "Orbital.hpp"
#include "Shader.hpp"
#include "Camera.hpp"
struct UserData
{
glm::mat4* projectionMatrix;
Camera* camera;
float frametime;
double lastX, lastY;
bool mouseMovedBefore;
bool cursorEnabled;
};
void OnFramebufferResize(GLFWwindow* window, int width, int height);
void OnMouseMoved(GLFWwindow* window, double xpos, double ypos);
void OnKeyPressed(GLFWwindow* window, int key, int scancode, int action, int mode);
void ProcessInput(GLFWwindow* window);
int main(int argc, char** argv)
{
2021-11-18 14:41:56 +00:00
// Initialize GLFW and let it know what OpenGL version/profile we're using
2021-11-16 21:57:55 +00:00
glfwInit();
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 4);
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 6);
glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
2021-11-18 14:41:56 +00:00
// Create a new window
// TODO: Magic numbers kinda bad
2021-11-16 21:57:55 +00:00
GLFWwindow* window = glfwCreateWindow(1200, 800, "Orbital Demonstation", NULL, NULL);
if (window == NULL)
{
std::cerr << "Failed to create GLFW window" << std::endl;
return -1;
}
2021-11-18 14:41:56 +00:00
// Make window context and set callbacks for resizing and keyboard/mouse events
2021-11-16 21:57:55 +00:00
glfwMakeContextCurrent(window);
glfwSetFramebufferSizeCallback(window, OnFramebufferResize);
glfwSetCursorPosCallback(window, OnMouseMoved);
glfwSetKeyCallback(window, OnKeyPressed);
2021-11-18 14:41:56 +00:00
// Let GLFW trap the cursor (needed for proper camera movement)
2021-11-16 21:57:55 +00:00
glfwSetInputMode(window, GLFW_CURSOR, GLFW_CURSOR_DISABLED);
2021-11-18 14:41:56 +00:00
// Load OpenGL functions from driver
2021-11-16 21:57:55 +00:00
if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress))
{
std::cerr << "Failed to initialize GLAD" << std::endl;
return -1;
}
2021-11-18 14:41:56 +00:00
// Set up ImGui
IMGUI_CHECKVERSION();
ImGui::CreateContext();
ImGuiIO& io = ImGui::GetIO();
ImGui_ImplGlfw_InitForOpenGL(window, true);
ImGui_ImplOpenGL3_Init("#version 460 core");
ImGui::StyleColorsDark();
// Create some orbital and set up its transformation matrix
// TODO: the matrix should probably be part of Model
2021-11-18 14:16:31 +00:00
Orbital orbital(2, 1);
2021-11-16 21:57:55 +00:00
glm::mat4 modelMatrix = glm::mat4(1.0f);
modelMatrix = glm::rotate(modelMatrix, glm::radians(90.0f), glm::vec3(1.0f, 0.0f, 0.0f));
modelMatrix = glm::scale(modelMatrix, glm::vec3(3.0f));
2021-11-18 14:41:56 +00:00
// Set up a camera
// TODO: should the projection matrix be part of the camera?
2021-11-16 21:57:55 +00:00
Camera camera;
camera.SetPosition(glm::vec3(0.0f, 0.0f, 3.0f));
glm::mat4 projectionMatrix = glm::perspective(glm::radians(110.0f), 1200.0f / 800.0f, 0.1f, 100.0f);
2021-11-18 14:41:56 +00:00
// Write some shaders to display the orbitals (too lazy to put them in files)
2021-11-16 21:57:55 +00:00
Shader shader(
R"(
#version 460 core
layout(location = 0) in vec3 position;
layout(location = 1) in vec3 color;
out vec3 outColor;
uniform mat4 model;
uniform mat4 view;
uniform mat4 projection;
void main()
{
outColor = color;
gl_Position = projection * view * model * vec4(position, 1.0f);
}
)",
R"(
#version 460 core
in vec3 outColor;
out vec4 FragColor;
void main()
{
FragColor = vec4(outColor, 1.0f);
}
)"
);
2021-11-18 14:41:56 +00:00
// Data that we want to be able to access from anywhere
2021-11-16 21:57:55 +00:00
UserData data = {
2021-11-18 14:41:56 +00:00
&projectionMatrix, // Projection matrix
&camera, // The camera object
0.0, // Duration of the last frame
0.0, 0.0, // Mouse position of the last frame
false, // Has the mouse moved before
false // Is the cursor enabled
2021-11-16 21:57:55 +00:00
};
glfwSetWindowUserPointer(window, &data);
2021-11-18 14:41:56 +00:00
// Set up a timer to calculate frametimes
2021-11-16 21:57:55 +00:00
std::chrono::system_clock::time_point start = std::chrono::system_clock::now();
2021-11-18 14:41:56 +00:00
// Set viewport and depth buffer
2021-11-16 21:57:55 +00:00
glViewport(0, 0, 1200, 800);
glEnable(GL_DEPTH_TEST);
while (!glfwWindowShouldClose(window))
{
2021-11-18 14:41:56 +00:00
// Calculate frametime
2021-11-16 21:57:55 +00:00
std::chrono::duration<float> framedelta = std::chrono::duration_cast<std::chrono::duration<float>>(std::chrono::system_clock::now() - start);
data.frametime = framedelta.count();
start = std::chrono::system_clock::now();
2021-11-18 14:41:56 +00:00
// Handle events
2021-11-16 21:57:55 +00:00
glfwPollEvents();
ProcessInput(window);
2021-11-18 14:41:56 +00:00
// Clear screen
glClearColor(0.0f, 0.0f, 0.05f, 0.0f);
2021-11-16 21:57:55 +00:00
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
2021-11-18 14:41:56 +00:00
// Start new ImGui Frame
ImGui_ImplOpenGL3_NewFrame();
ImGui_ImplGlfw_NewFrame();
ImGui::NewFrame();
// Use our shader and set the matrices
2021-11-16 21:57:55 +00:00
shader.Bind();
shader.SetMatrix("model", &modelMatrix[0][0]);
shader.SetMatrix("view", camera.GetViewMatrix());
shader.SetMatrix("projection", &projectionMatrix[0][0]);
2021-11-18 14:41:56 +00:00
// Finally draw the orbital
2021-11-16 21:57:55 +00:00
orbital.Draw();
2021-11-18 14:41:56 +00:00
ImGui::Begin("Test Window");
ImGui::Button("Test");
ImGui::End();
ImGui::Render();
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
// Update swapchain
2021-11-16 21:57:55 +00:00
glfwSwapBuffers(window);
}
2021-11-18 14:41:56 +00:00
// cleanup
ImGui_ImplOpenGL3_Shutdown();
ImGui_ImplGlfw_Shutdown();
ImGui::DestroyContext();
2021-11-16 21:57:55 +00:00
glfwDestroyWindow(window);
return 0;
}
void OnFramebufferResize(GLFWwindow* window, int width, int height)
{
glViewport(0, 0, width, height);
UserData* data = (UserData*)glfwGetWindowUserPointer(window);
*(data->projectionMatrix) = glm::perspective(glm::radians(110.0f), (float)width / (float)height, 0.1f, 100.0f);
}
void OnMouseMoved(GLFWwindow* window, double xpos, double ypos)
{
UserData* data = (UserData*)glfwGetWindowUserPointer(window);
2021-11-18 14:41:56 +00:00
float sensitivity = 6.5f;
2021-11-16 21:57:55 +00:00
if (!data->mouseMovedBefore)
{
data->lastX = xpos;
data->lastY = ypos;
data->mouseMovedBefore = true;
}
double deltaX = xpos - data->lastX;
double deltaY = data->lastY - ypos;
data->lastX = xpos;
data->lastY = ypos;
if (!data->cursorEnabled)
{
data->camera->HandleMouseMoved(deltaX, deltaY, sensitivity, data->frametime);
}
}
void OnKeyPressed(GLFWwindow* window, int key, int scancode, int action, int mode)
{
UserData* data = (UserData*)glfwGetWindowUserPointer(window);
if (key == GLFW_KEY_ESCAPE && action == GLFW_PRESS)
{
data->cursorEnabled = !data->cursorEnabled;
glfwSetInputMode(window, GLFW_CURSOR, (data->cursorEnabled ? GLFW_CURSOR_NORMAL : GLFW_CURSOR_DISABLED));
}
}
void ProcessInput(GLFWwindow* window)
{
UserData* data = (UserData*)glfwGetWindowUserPointer(window);
float cameraSpeed = 3.0f;
if (glfwGetKey(window, GLFW_KEY_W) == GLFW_PRESS)
data->camera->MoveForward(cameraSpeed, data->frametime);
if (glfwGetKey(window, GLFW_KEY_S) == GLFW_PRESS)
data->camera->MoveForward(-cameraSpeed, data->frametime);
if (glfwGetKey(window, GLFW_KEY_D) == GLFW_PRESS)
data->camera->MoveRight(cameraSpeed, data->frametime);
if (glfwGetKey(window, GLFW_KEY_A) == GLFW_PRESS)
data->camera->MoveRight(-cameraSpeed, data->frametime);
if (glfwGetKey(window, GLFW_KEY_SPACE) == GLFW_PRESS)
data->camera->MoveUp(cameraSpeed, data->frametime);
if (glfwGetKey(window, GLFW_KEY_LEFT_SHIFT) == GLFW_PRESS)
data->camera->MoveUp(-cameraSpeed, data->frametime);
}