From b18542947546fb143b407fc6cfa0874c48e720f7 Mon Sep 17 00:00:00 2001 From: Robert Date: Fri, 29 Jan 2021 20:36:17 +0100 Subject: [PATCH] Added spotlight --- examples/movement/main.cpp | 45 ++++++++++++++++++- examples/movement/shaders/fragmentShader.frag | 30 +++++++++---- include/camera.hpp | 2 + include/lighting/point.hpp | 4 +- include/lighting/spotlight.hpp | 27 +++++++++++ include/openglu.hpp | 1 + src/camera.cpp | 5 +++ src/lighting/point.cpp | 8 +++- src/lighting/spotlight.cpp | 28 ++++++++++++ 9 files changed, 139 insertions(+), 11 deletions(-) create mode 100644 include/lighting/spotlight.hpp create mode 100644 src/lighting/spotlight.cpp diff --git a/examples/movement/main.cpp b/examples/movement/main.cpp index 808a5d6..d8dc5d3 100644 --- a/examples/movement/main.cpp +++ b/examples/movement/main.cpp @@ -234,6 +234,10 @@ int main(int argc, char** argv) oglu::PointLight pointLight; pointLight.LinkPositionToTransformable(lightSource); + oglu::SpotLight flashlight; + flashlight.linear = 0.09f; + flashlight.quadratic = 0.032f; + // Create a shader oglu::Shader shader, lightSourceShader; try @@ -270,15 +274,31 @@ int main(int argc, char** argv) ImGui_ImplGlfw_NewFrame(); ImGui::NewFrame(); + flashlight.SetPosition(camera.GetPosition()); + flashlight.direction = camera.GetFront(); + shader->Use(); + shader->SetUniform("light.ambient", "light.ambientStrength", ambient); + /* shader->SetUniform3fv("light.position", 1, glm::value_ptr(lightSource.GetPosition())); shader->SetUniform("light.diffuse", pointLight.diffusionColor, true); shader->SetUniform("light.specular", pointLight.specularColor, true); shader->SetUniform("light.constant", pointLight.constant); shader->SetUniform("light.linear", pointLight.linear); shader->SetUniform("light.quadratic", pointLight.quadratic); + */ + + shader->SetUniform3fv("fl.position", 1, glm::value_ptr(camera.GetPosition())); + shader->SetUniform3fv("fl.direction", 1, glm::value_ptr(flashlight.direction)); + shader->SetUniform("fl.angle", glm::cos(glm::radians(flashlight.angle))); + shader->SetUniform("fl.outerAngle", glm::cos(glm::radians(flashlight.outerAngle))); + shader->SetUniform("fl.diffuse", flashlight.diffusionColor, true); + shader->SetUniform("fl.specular", flashlight.specularColor, true); + shader->SetUniform("fl.constant", flashlight.constant); + shader->SetUniform("fl.linear", flashlight.linear); + shader->SetUniform("fl.quadratic", flashlight.quadratic); shader->SetUniform3fv("viewPos", 1, glm::value_ptr(camera.GetPosition())); @@ -297,12 +317,14 @@ int main(int argc, char** argv) cube.Render(); } + /* lightSourceShader->Use(); lightSourceShader->SetUniformMatrix4fv("model", 1, GL_FALSE, glm::value_ptr(lightSource.GetMatrix(true))); lightSourceShader->SetUniformMatrix4fv("view", 1, GL_FALSE, glm::value_ptr(camera.GetMatrix())); lightSourceShader->SetUniformMatrix4fv("projection", 1, GL_FALSE, glm::value_ptr(camera.GetProjection())); lightSourceShader->SetUniform("color", pointLight.diffusionColor, true); lightSource.Render(); + */ ImGui::Begin("Controls"); @@ -328,7 +350,7 @@ int main(int argc, char** argv) ImGui::Separator(); } - ImGui::SetNextItemOpen(true, ImGuiCond_Once); + ImGui::SetNextItemOpen(false, ImGuiCond_Once); if (ImGui::TreeNode("Point")) { ImGui::ColorEdit3("Diffusion", &pointLight.diffusionColor.r); @@ -347,6 +369,27 @@ int main(int argc, char** argv) ImGui::TreePop(); } + + ImGui::SetNextItemOpen(true, ImGuiCond_Once); + if (ImGui::TreeNode("Flashlight")) + { + ImGui::ColorEdit3("Diffusion", &flashlight.diffusionColor.r); + ImGui::ColorEdit3("Specular", &flashlight.specularColor.r); + ImGui::SliderFloat("Angle", &flashlight.angle, 1.0f, flashlight.outerAngle - 1.0f); + ImGui::SliderFloat("Outer angle", &flashlight.outerAngle, flashlight.angle + 1.0f, 60.0f); + + ImGui::SetNextItemOpen(true, ImGuiCond_Once); + if (ImGui::TreeNode("Attenuation")) + { + ImGui::SliderFloat("Constant", &flashlight.constant, 1.0f, 4.0f); + ImGui::SliderFloat("Linear", &flashlight.linear, 0.0014f, 0.7f); + ImGui::SliderFloat("Quadratic", &flashlight.quadratic, 0.00007f, 1.8f); + + ImGui::TreePop(); + } + + ImGui::TreePop(); + } } ImGui::SetNextItemOpen(true, ImGuiCond_Once); diff --git a/examples/movement/shaders/fragmentShader.frag b/examples/movement/shaders/fragmentShader.frag index 1abbdc0..8621997 100644 --- a/examples/movement/shaders/fragmentShader.frag +++ b/examples/movement/shaders/fragmentShader.frag @@ -15,6 +15,15 @@ struct Light float constant, linear, quadratic; }; +struct Flashlight +{ + vec3 position, direction; + float angle, outerAngle; + vec3 diffuse, specular; + + float constant, linear, quadratic; +}; + in vec2 oUV; in vec3 oNormal; in vec3 oFragPos; @@ -25,28 +34,33 @@ uniform vec3 viewPos; uniform Material material; uniform Light light; +uniform Flashlight fl; void main() { - // Ambient light - vec3 ambient = light.ambient * light.ambientStrength * vec3(texture(material.diffuse, oUV)); + vec3 lightDir = normalize(fl.position - oFragPos); + float theta = dot(lightDir, normalize(-fl.direction)); + float epsilon = fl.angle - fl.outerAngle; + float intensity = clamp((theta - fl.outerAngle) / epsilon, 0.0, 1.0); vec3 norm = normalize(oNormal); // Diffuse light - vec3 lightDir = normalize(light.position - oFragPos); float diff = max(dot(norm, lightDir), 0.0); - vec3 diffuse = light.diffuse * diff * vec3(texture(material.diffuse, oUV)); + vec3 diffuse = fl.diffuse * diff * vec3(texture(material.diffuse, oUV)); // Specular light vec3 viewDir = normalize(viewPos - oFragPos); vec3 reflectDir = reflect(-lightDir, norm); float spec = pow(max(dot(viewDir, reflectDir), 0.0), material.shininess); - vec3 specular = light.specular * spec * vec3(texture(material.specular, oUV)); + vec3 specular = fl.specular * spec * vec3(texture(material.specular, oUV)); - float dist = length(light.position - oFragPos); - float attenuation = 1.0 / (light.constant + light.linear * dist + light.quadratic * (dist * dist)); + float dist = length(fl.position - oFragPos); + float attenuation = 1.0 / (fl.constant + fl.linear * dist + fl.quadratic * (dist * dist)); - FragColor = vec4((ambient + diffuse + specular) * attenuation, 1.0); + // Ambient light + vec3 ambient = light.ambient * light.ambientStrength * vec3(texture(material.diffuse, oUV)); + + FragColor = vec4(ambient + ((diffuse + specular) * attenuation * intensity), 1.0); } \ No newline at end of file diff --git a/include/camera.hpp b/include/camera.hpp index 88dfd5e..5c8a182 100644 --- a/include/camera.hpp +++ b/include/camera.hpp @@ -210,6 +210,8 @@ namespace oglu */ const glm::mat4& GetProjection(); + const glm::vec3& GetFront(); + public: float fov; ///< FOV of the camera float aspectRatio; ///< Aspect ratio of the camera diff --git a/include/lighting/point.hpp b/include/lighting/point.hpp index 4a53a14..a095b8a 100644 --- a/include/lighting/point.hpp +++ b/include/lighting/point.hpp @@ -75,6 +75,8 @@ namespace oglu */ float* GetPositionPointer(); + void SetPosition(const glm::vec3& position); + public: // TODO: Does this split make sense? I'll keep it for now Color diffusionColor; ///< Diffusion color of the light @@ -84,7 +86,7 @@ namespace oglu float linear; ///< Coefficient of the linear attenuation term float quadratic; ///< Coefficient of the quadratic attenuation term - private: + protected: glm::vec3* position; ///< Position of the light bool isLinked; ///< Wether the position is linked to an object or not }; diff --git a/include/lighting/spotlight.hpp b/include/lighting/spotlight.hpp new file mode 100644 index 0000000..0c1ba1e --- /dev/null +++ b/include/lighting/spotlight.hpp @@ -0,0 +1,27 @@ +#ifndef SPOTLIGHT_HPP +#define SPOTLIGHT_HPP + +#include +#include + +#include + +namespace oglu +{ + class OGLU_API SpotLight : public PointLight + { + public: + SpotLight(); + SpotLight(const glm::vec3& position, const glm::vec3& direction, float angle, const Color& diffusionColor = Color::White, const Color& specularColor = Color::White); + SpotLight(const SpotLight& other); + + float angle; + float outerAngle; + glm::vec3 direction; + + private: + + }; +} + +#endif \ No newline at end of file diff --git a/include/openglu.hpp b/include/openglu.hpp index 3876b27..994ccea 100644 --- a/include/openglu.hpp +++ b/include/openglu.hpp @@ -18,6 +18,7 @@ #include #include +#include namespace oglu { diff --git a/src/camera.cpp b/src/camera.cpp index 7613223..4abf4ee 100644 --- a/src/camera.cpp +++ b/src/camera.cpp @@ -150,4 +150,9 @@ namespace oglu { return projection = glm::perspective(glm::radians(fov), aspectRatio, zNear, zFar);; } + + const glm::vec3& Camera::GetFront() + { + return front; + } } diff --git a/src/lighting/point.cpp b/src/lighting/point.cpp index 1c81182..039e60b 100644 --- a/src/lighting/point.cpp +++ b/src/lighting/point.cpp @@ -21,7 +21,7 @@ namespace oglu position(new glm::vec3(0.0f)), diffusionColor(other.diffusionColor), specularColor(other.specularColor), constant(1.0f), linear(0.09f), quadratic(0.032f), isLinked(false) { - memcpy(this->position, position, sizeof(glm::vec3)); + memcpy(this->position, other.position, sizeof(glm::vec3)); } PointLight::~PointLight() @@ -51,4 +51,10 @@ namespace oglu { return &((*position)[0]); } + + void PointLight::SetPosition(const glm::vec3& position) + { + if (!isLinked) + memcpy(this->position, &position, sizeof(glm::vec3)); + } } diff --git a/src/lighting/spotlight.cpp b/src/lighting/spotlight.cpp new file mode 100644 index 0000000..5061f06 --- /dev/null +++ b/src/lighting/spotlight.cpp @@ -0,0 +1,28 @@ +#include "lighting/spotlight.hpp" + +namespace oglu +{ + SpotLight::SpotLight() : + PointLight(), direction(glm::vec3(0.0f, -1.0f, 0.0f)), angle(18.5f), outerAngle(25.0f) + { + } + + SpotLight::SpotLight(const glm::vec3& position, const glm::vec3& direction, float angle, const Color& diffusionColor, const Color& specularColor) : + PointLight(position, diffusionColor, specularColor), direction(glm::vec3(0.0f, -1.0f, 0.0f)), angle(18.5f), outerAngle(25.0f) + { + } + + SpotLight::SpotLight(const SpotLight& other) : + direction(glm::vec3(0.0f, -1.0f, 0.0f)), angle(18.5f), outerAngle(25.0f) + { + diffusionColor = other.diffusionColor; + specularColor = other.specularColor; + constant = other.constant; + linear = other.linear; + quadratic = other.quadratic; + isLinked = false; + + position = new glm::vec3(0.0f); + memcpy(this->position, other.position, sizeof(glm::vec3)); + } +}