Camera movement

This commit is contained in:
Robert 2020-09-06 13:37:17 +02:00
parent dfce02cd1c
commit 419ab8261d
3 changed files with 122 additions and 28 deletions

View file

@ -1,5 +1,6 @@
#include <iostream>
#include <vector>
#include <functional>
#include <glad/glad.h>
#include <GLFW/glfw3.h>
@ -20,6 +21,10 @@
int width = 800;
int height = 800;
bool trapCursor = true;
std::function<void(GLFWwindow*, float, float)> handleMouseCallback;
void LogGlfwError(const char* message)
{
const char* error;
@ -56,6 +61,63 @@ void FramebufferSizeCallback(GLFWwindow* window, int w, int h)
glViewport(0, 0, w, h);
}
void CursorPosCallback(GLFWwindow* window, double x, double y)
{
if (trapCursor)
handleMouseCallback(window, x, y);
}
void KeyCallback(GLFWwindow* window, int key, int mode, int action, int mods)
{
if (key == GLFW_KEY_ESCAPE && action == GLFW_PRESS)
{
trapCursor = !trapCursor;
glfwSetInputMode(window, GLFW_CURSOR, (trapCursor ? GLFW_CURSOR_DISABLED : GLFW_CURSOR_NORMAL));
glfwSetCursorPos(window, width / 2, height / 2);
}
}
void HandleKeyboard(GLFWwindow* window, Camera& cam, float frametime)
{
static float movementSpeed = 4.5f;
// Movement
if (glfwGetKey(window, GLFW_KEY_W) == GLFW_PRESS)
cam.Move(cam.front * movementSpeed * frametime);
if (glfwGetKey(window, GLFW_KEY_S) == GLFW_PRESS)
cam.Move(-cam.front * movementSpeed * frametime);
if (glfwGetKey(window, GLFW_KEY_A) == GLFW_PRESS)
cam.Move(-cam.right * movementSpeed * frametime);
if (glfwGetKey(window, GLFW_KEY_D) == GLFW_PRESS)
cam.Move(cam.right * movementSpeed * frametime);
if (glfwGetKey(window, GLFW_KEY_SPACE) == GLFW_PRESS)
cam.Move(glm::vec3(0.0f, 1.0f, 0.0f) * movementSpeed * frametime);
if (glfwGetKey(window, GLFW_KEY_LEFT_SHIFT) == GLFW_PRESS)
cam.Move(-glm::vec3(0.0f, 1.0f, 0.0f) * movementSpeed * frametime);
}
void HandleMouse(GLFWwindow* window, Camera* cam, float x, float y)
{
static const float sensitivity = 0.1f;
static float pitch = 0.0f;
static float yaw = -90.0f;
pitch += (height / 2 - y) * sensitivity;
yaw += (x - width / 2) * sensitivity;
if (pitch > 89.0f)
pitch = 89.0f;
else if (pitch < -89.0f)
pitch = -89.0f;
cam->SetRotation(glm::vec2(pitch, yaw));
glfwSetCursorPos(window, width / 2.0f, height / 2.0f);
}
int main(int argc, char** argv)
{
// Initialize ImGui
@ -135,18 +197,27 @@ int main(int argc, char** argv)
}
// A camera
Camera cam(glm::vec3(0.0f, 0.0f, 10.0f), glm::vec3(0.0f, 0.0f, 0.0f));
Camera cam(glm::vec3(0.0f, 0.0f, 10.0f), glm::vec2(0.0f, -90.0f));
glfwSetInputMode(window, GLFW_CURSOR, GLFW_CURSOR_DISABLED);
glfwSetCursorPos(window, width / 2, height / 2);
handleMouseCallback = std::bind(HandleMouse, std::placeholders::_1, &cam, std::placeholders::_2, std::placeholders::_3);
glfwSetCursorPosCallback(window, CursorPosCallback);
glfwSetKeyCallback(window, KeyCallback);
// So that stuff isnt clipping through each other
glEnable(GL_DEPTH_TEST);
float lastTime = glfwGetTime();
while (!glfwWindowShouldClose(window))
{
glfwPollEvents();
float currentTime = glfwGetTime();
HandleKeyboard(window, cam, currentTime - lastTime);
lastTime = currentTime;
float t = glfwGetTime();
static float PI = glm::pi<float>();
@ -175,12 +246,6 @@ int main(int argc, char** argv)
ImGui::SliderFloat("zFar", &zFar, 10.0f, 100.0f);
}
if (ImGui::CollapsingHeader("Orbit"))
{
ImGui::SliderFloat("Speed", &orbitSpeed, 0.0f, 4 * PI, "%.2f rad/s");
ImGui::SliderFloat("Radius", &orbitRadius, 1.0f, 50.0f, "%.1f u");
}
ImGui::End();
perspective = glm::perspective(fov, (float)width/(float)height, zNear, zFar);
@ -188,8 +253,6 @@ int main(int argc, char** argv)
cubeTexture.Bind();
cubeShader.Use();
cubeShader.SetUniformMat4("projection", &perspective[0][0]);
cam.SetPosition(orbitRadius * glm::vec3(cos(orbitSpeed * t), 0.0f, sin(orbitSpeed * t)));
cam.Use(cubeShader);
for (Cube* cube : cubes)

View file

@ -2,8 +2,20 @@
#include <glm/gtc/matrix_transform.hpp>
Camera::Camera(glm::vec3 position, glm::vec3 target) :
position(position), target(target)
Camera::Camera(glm::vec3 position, glm::vec3 direction) :
position(position), direction(direction)
{
CalculateCamera();
}
Camera::Camera(glm::vec3 position, glm::vec2 pitchYaw) :
position(position), direction(
glm::vec3(
cos(glm::radians(pitchYaw.y)) * cos(glm::radians(pitchYaw.x)),
sin(glm::radians(pitchYaw.x)),
sin(glm::radians(pitchYaw.y)) * cos(glm::radians(pitchYaw.x))
)
)
{
CalculateCamera();
}
@ -14,18 +26,32 @@ void Camera::SetPosition(glm::vec3 newPos)
CalculateCamera();
}
void Camera::SetTarget(glm::vec3 newTarget)
{
target = newTarget;
CalculateCamera();
}
void Camera::Move(glm::vec3 movement)
{
position += movement;
CalculateCamera();
}
void Camera::SetRotation(glm::vec2 pitchYaw)
{
direction = glm::vec3(
cos(glm::radians(pitchYaw.y)) * cos(glm::radians(pitchYaw.x)),
sin(glm::radians(pitchYaw.x)),
sin(glm::radians(pitchYaw.y)) * cos(glm::radians(pitchYaw.x))
);
CalculateCamera();
}
void Camera::Rotate(glm::vec2 pitchYaw)
{
direction += glm::vec3(
cos(glm::radians(pitchYaw.y)) * cos(glm::radians(pitchYaw.x)),
sin(glm::radians(pitchYaw.x)),
sin(glm::radians(pitchYaw.y)) * cos(glm::radians(pitchYaw.x))
);
CalculateCamera();
}
void Camera::Use(const Shader& program)
{
program.SetUniformMat4("view", &viewMatrix[0][0]);
@ -33,9 +59,10 @@ void Camera::Use(const Shader& program)
void Camera::CalculateCamera()
{
direction = glm::normalize(target - position);
right = glm::normalize(glm::cross(glm::vec3(0.0f, 1.0f, 0.0f), direction));
up = glm::cross(direction, right);
front = glm::normalize(direction);
right = glm::normalize(glm::cross(front, glm::vec3(0.0f, 1.0f, 0.0f)));
// right = glm::normalize(glm::cross(glm::vec3(0.0f, 1.0f, 0.0f), front));
up = glm::cross(right, front);
viewMatrix = glm::lookAt(position, target, up);
viewMatrix = glm::lookAt(position, position + front, up);
}

View file

@ -8,20 +8,24 @@ class Camera
{
public:
Camera() = default;
Camera(glm::vec3 position, glm::vec3 target);
Camera(glm::vec3 position, glm::vec3 direction);
Camera(glm::vec3 position, glm::vec2 pitchYaw);
void SetPosition(glm::vec3 newPos);
void SetTarget(glm::vec3 newTarget);
void Move(glm::vec3 movement);
void SetRotation(glm::vec2 pitchYaw);
void Rotate(glm::vec2 pitchYaw);
void Use(const Shader& program);
public:
glm::vec3 front, up, right;
private:
void CalculateCamera();
glm::vec3 position, target;
glm::vec3 direction, up, right;
glm::vec3 position, direction;
glm::mat4 viewMatrix;
};