From bf751c6c696371c9c3c727f4dd0196f413164afe Mon Sep 17 00:00:00 2001 From: Robert Date: Fri, 29 Jan 2021 17:28:48 +0100 Subject: [PATCH] Reworked material --- CMakeLists.txt | 6 +-- examples/movement/main.cpp | 22 +++++---- include/core.hpp | 28 +++++++++++- include/lighting/ambient.hpp | 41 ++++++++++++++++- include/lighting/point.hpp | 62 ++++++++++++++++++++++++-- include/material.hpp | 86 ++++++++++++++++++++++++++++++++++++ include/object.hpp | 13 +----- include/openglu.hpp | 1 + src/core.cpp | 6 +++ src/lighting/ambient.cpp | 2 +- src/material.cpp | 14 ++++++ src/object.cpp | 2 + 12 files changed, 251 insertions(+), 32 deletions(-) create mode 100644 include/material.hpp create mode 100644 src/core.cpp create mode 100644 src/material.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index a5c2fe7..448cdd1 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -4,7 +4,7 @@ project(OpenGLUtility) set(CMAKE_INSTALL_RPATH "${CMAKE_INSTALL_PREFIX}/lib") set(CMAKE_INSTALL_RPATH_USE_LINK_PATH TRUE) -# set(CMAKE_CXX_STANDARD 17) +set(CMAKE_CXX_STANDARD 17) set(BUILD_EXAMPLES ON CACHE BOOL "Build examples") set(BUILD_DOCUMENTATION ON CACHE BOOL "Generate documentation") @@ -33,13 +33,11 @@ file(GLOB_RECURSE source_files "vendor/src/*.c" ) -message(STATUS ${source_files}) - add_library(openglu SHARED ${include_files} ${source_files} "cpp.hint" -) + "src/core.cpp") target_compile_definitions(openglu PRIVATE OGLU_BUILD_DLL) diff --git a/examples/movement/main.cpp b/examples/movement/main.cpp index 65d041d..cf75cb2 100644 --- a/examples/movement/main.cpp +++ b/examples/movement/main.cpp @@ -184,6 +184,12 @@ int main(int argc, char** argv) // Make a square oglu::VertexArray cubeDefault = oglu::MakeVertexArray(vertices, sizeof(vertices), nullptr, 0, topology, sizeof(topology)); oglu::SharedMaterial cubeMaterial(new oglu::Material); + + //cubeMaterial->AddProperty("ambient", oglu::Color::White); + cubeMaterial->AddProperty("diffuse", oglu::Color::White); + cubeMaterial->AddProperty("specular", oglu::Color::White); + cubeMaterial->AddProperty("shininess", 32.f); + oglu::Object cubes[10] = { oglu::Object(cubeDefault), oglu::Object(cubeDefault), @@ -287,10 +293,10 @@ int main(int argc, char** argv) shader->SetUniform("model", cube); shader->SetUniformMatrix3fv("normal", 1, GL_FALSE, glm::value_ptr(cube.GetNormalMatrix())); - shader->SetUniform("material.ambient", cube.material->ambient, true); - shader->SetUniform("material.diffuse", cube.material->diffuse, true); - shader->SetUniform("material.specular", cube.material->specular, true); - shader->SetUniform("material.shininess", cube.material->shininess); + shader->SetUniform("material.ambient", cube.material->GetPropertyValue("ambient"), true); + shader->SetUniform("material.diffuse", cube.material->GetPropertyValue("diffuse"), true); + shader->SetUniform("material.specular", cube.material->GetPropertyValue("specular"), true); + shader->SetUniform("material.shininess", cube.material->GetPropertyValue("shininess")); cube.Render(); } @@ -339,10 +345,10 @@ int main(int argc, char** argv) ImGui::SetNextItemOpen(true, ImGuiCond_Once); if (ImGui::CollapsingHeader("Cube Material")) { - ImGui::ColorEdit3("Ambient", &(cubeMaterial->ambient.r)); - ImGui::ColorEdit3("Diffuse", &(cubeMaterial->diffuse.r)); - ImGui::ColorEdit3("Specular", &(cubeMaterial->specular.r)); - ImGui::SliderFloat("Shininess", &(cubeMaterial->shininess), 1.0f, 256.0f); + ImGui::ColorEdit3("Ambient", &(cubeMaterial->GetProperty("ambient")->r)); + ImGui::ColorEdit3("Diffuse", &(cubeMaterial->GetProperty("diffuse")->r)); + ImGui::ColorEdit3("Specular", &(cubeMaterial->GetProperty("specular")->r)); + ImGui::SliderFloat("Shininess", cubeMaterial->GetProperty("shininess"), 1.0f, 256.0f); } ImGui::End(); diff --git a/include/core.hpp b/include/core.hpp index 3909ced..bb67b5e 100644 --- a/include/core.hpp +++ b/include/core.hpp @@ -12,12 +12,12 @@ #pragma warning(disable : 4251) #include +#include #include #include #include - #ifdef OGLU_WIN32 #ifdef OGLU_BUILD_DLL #define OGLU_API __declspec(dllexport) @@ -28,4 +28,30 @@ #define OGLU_API #endif //OGLU_WIN32 +namespace oglu +{ + class NullBuffer : public std::streambuf + { + public: + int overflow(int c) { return c; } + }; + + class NullStream : public std::ostream + { + public: + NullStream() : std::ostream(&buf) {} + + private: + NullBuffer buf; + }; + + extern OGLU_API NullStream cnull; +} + +#ifndef NDEBUG + #define OGLU_ERROR_STREAM std::cerr +#else + #define OGLU_ERROR_STREAM oglu::cnull +#endif + #endif diff --git a/include/lighting/ambient.hpp b/include/lighting/ambient.hpp index a4c9cd3..42f42d2 100644 --- a/include/lighting/ambient.hpp +++ b/include/lighting/ambient.hpp @@ -1,3 +1,10 @@ +/*****************************************************************//** + * @file ambient.hpp + * @brief Anything related to ambient/environmental lighting + * + * @author Lauchmelder + * @date January 2021 + *********************************************************************/ #ifndef AMBIENT_HPP #define AMBIENT_HPP @@ -6,16 +13,46 @@ namespace oglu { + /** + * @brief A class holding information needed for ambient lighting. + */ class OGLU_API AmbientLight { public: + /** + * @brief Create default ambient light. + * + * By default, this creates a white light at full intensity + */ AmbientLight(); + + /** + * @brief Create a custom ambient light. + * + * @param r Red component + * @param r Green component + * @param r Blue component + * @param intensity Intensity of the lighting + */ AmbientLight(GLfloat r, GLfloat g, GLfloat b, GLfloat intensity); + + /** + * @brief Create a custom ambient light. + * + * @param color Color of the lighting + * @param intensity Intensity of the lighting + */ AmbientLight(const Color& color, GLfloat intensity); + + /** + * @brief Copy another ambient light. + * + * @param other The light to copy from + */ AmbientLight(const AmbientLight& other); - GLfloat intensity; - Color color; + GLfloat intensity; ///< Intensity of the ambient light + Color color; ///< Color of the ambient light }; } diff --git a/include/lighting/point.hpp b/include/lighting/point.hpp index b43d2fa..9f69735 100644 --- a/include/lighting/point.hpp +++ b/include/lighting/point.hpp @@ -1,3 +1,10 @@ +/*****************************************************************//** + * @file point.hpp + * @brief Anything related to point light sources + * + * @author Lauchmelder + * @date January 2021 + *********************************************************************/ #ifndef POINT_HPP #define POINT_HPP @@ -9,27 +16,74 @@ namespace oglu { class Transformable; + /** + * @brief A class containing things related to point lighting. + */ class OGLU_API PointLight { public: + /** + * @brief Create a default point light. + * + * By default the diffusion and specular components are white and + * the light is positioned at the world origin + */ PointLight(); + + /** + * @brief Create a new point light. + * + * @param position Position of the light + * @param diffusionColor Color of the diffusion component + * @param specularColor Color of the specular component + */ PointLight(const glm::vec3& position, const Color& diffusionColor = Color::White, const Color& specularColor = Color::White); + + /** + * @brief Copy from another point light. + * + * @param other The point light to copy from + */ PointLight(const PointLight& other); ~PointLight(); + /** + * @brief Link the lights position to a Transformable. + * + * It is possible to link the positions of a Transformable and + * a PointLight. This means that these two objects will always + * have the same position, until they are unlinked. Deleting a + * Transformable before unlinking any linked PointLight causes + * undefined behaviour. + * + * @param link The Transformable to link to + */ void LinkPositionToTransformable(Transformable& link); + + /** + * @brief Unlink a Transformable. + * + * This breaks the link between the two objects. + */ void UnlinkPositionFromTransformable(); + + /** + * Get the lights position. + * + * @returns A pointer to the lights position + */ float* GetPositionPointer(); public: - Color diffusionColor; - Color specularColor; + // TODO: Does this split make sense? I'll keep it for now + Color diffusionColor; ///< Diffusion color of the light + Color specularColor; ///< Specular color of the light private: - glm::vec3* position; - bool isLinked; + glm::vec3* position; ///< Position of the light + bool isLinked; ///< Wether the position is linked to an object or not }; } diff --git a/include/material.hpp b/include/material.hpp new file mode 100644 index 0000000..de728e9 --- /dev/null +++ b/include/material.hpp @@ -0,0 +1,86 @@ +/*****************************************************************//** + * @file material.hpp + * @brief Anything related to materials + * + * @author Lauchmelder + * @date January 2021 + *********************************************************************/ +#ifndef MATERIAL_HPP +#define MATERIAL_HPP + +#include +#include +#include +#include + +namespace oglu +{ + /** + * @brief A structure representing an object's material. + * + * This object is dynamic and can contain any data that you want + */ + class OGLU_API Material + { + public: + /** + * @brief Adds a new material property. + * + * @param name The name of the property + * @param value The value of the property + */ + void AddProperty(const std::string& name, const std::any& value); + + /** + * @brief Remove the given property from the material. + */ + void RemoveProperty(const std::string& name); + + /** + * @brief Get a property from the material. + * + * Should the material property not be present in the material this function + * will attempt to add it, using the default constructor of the supplied class. + * You should not rely on this failsafe working properly all the time. + * + * @tparam T The type of the stores property + * @param name The name of the property + * @returns A pointer to the stored property + */ + template T* GetProperty(const std::string& name, std::ostream& errorStream = OGLU_ERROR_STREAM) + { + std::map::iterator it = properties.find(name); + if (it == properties.end()) + { + errorStream << "Failed to locate material property \"" << name << "\" in " << static_cast(this) << ". Trying to construct default property of type \"" << typeid(T).name() << "\" and add it to the material.\n"; + + it = properties.insert(std::make_pair(name, T())).first; + } + + return std::any_cast(&(it->second)); + } + + /** + * @brief Get a property from the material. + * + * Should the material property not be present in the material this function + * will attempt to add it, using the default constructor of the supplied class. + * You should not rely on this failsafe working properly all the time. + * + * @tparam T The type of the stores property + * @param name The name of the property + * @returns The value of the stored property + */ + template inline T GetPropertyValue(const std::string& name, std::ostream& errorStream = OGLU_ERROR_STREAM) + { + return *(GetProperty(name, errorStream)); + } + + private: + std::map properties; + }; + + typedef std::shared_ptr SharedMaterial; ///< A material that can safely be shared between objects +} + +#endif \ No newline at end of file diff --git a/include/object.hpp b/include/object.hpp index 6d3594a..022e4c9 100644 --- a/include/object.hpp +++ b/include/object.hpp @@ -15,18 +15,7 @@ namespace oglu { - /** - * @brief A structure representing an object's material. - */ - class OGLU_API Material - { - public: - Color ambient = Color::White; - Color diffuse = Color::White; - Color specular = Color::White; - float shininess = 32.0f; - }; - + class Material; typedef std::shared_ptr SharedMaterial; /** diff --git a/include/openglu.hpp b/include/openglu.hpp index e9ad54a..3876b27 100644 --- a/include/openglu.hpp +++ b/include/openglu.hpp @@ -13,6 +13,7 @@ #include #include #include +#include #include #include diff --git a/src/core.cpp b/src/core.cpp new file mode 100644 index 0000000..a61335b --- /dev/null +++ b/src/core.cpp @@ -0,0 +1,6 @@ +#include "core.hpp" + +namespace oglu +{ + NullStream cnull; +} \ No newline at end of file diff --git a/src/lighting/ambient.cpp b/src/lighting/ambient.cpp index f70fd76..93c8a5d 100644 --- a/src/lighting/ambient.cpp +++ b/src/lighting/ambient.cpp @@ -3,7 +3,7 @@ namespace oglu { AmbientLight::AmbientLight() : - color(1.f, 1.f, 1.f), intensity(1.0f) + color(Color::White), intensity(1.0f) { } diff --git a/src/material.cpp b/src/material.cpp new file mode 100644 index 0000000..e362cf7 --- /dev/null +++ b/src/material.cpp @@ -0,0 +1,14 @@ +#include "material.hpp" + +namespace oglu +{ + void Material::AddProperty(const std::string& name, const std::any& value) + { + properties.insert(std::make_pair(name, value)); + } + + void Material::RemoveProperty(const std::string& name) + { + properties.erase(name); + } +} diff --git a/src/object.cpp b/src/object.cpp index 2b867d5..752dd35 100644 --- a/src/object.cpp +++ b/src/object.cpp @@ -1,5 +1,7 @@ #include "object.hpp" +#include + namespace oglu { Object::Object(const GLfloat* vertices, size_t verticesSize, const GLuint* indices, size_t indicesSize, const VertexAttribute* topology, size_t topologySize) :