Compare commits
	
		
			22 Commits
		
	
	
		
			windows
			...
			8a044cbe86
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 8a044cbe86 | |||
| c253fff7df | |||
| 97b34962f7 | |||
| 620adb20f9 | |||
| 507ba483b3 | |||
| fc91f6662e | |||
| 2b0494a23d | |||
| fec93b098b | |||
| 4a9f45b6ef | |||
| fdb891d860 | |||
| 5bcfb18296 | |||
| 303b931fb7 | |||
| 69dbe5ae2f | |||
| de6496ff81 | |||
| 807e0ce9d9 | |||
| 58e25b530b | |||
| 39f528d7ad | |||
| 6dc269ce13 | |||
| 9d5bb51463 | |||
| c5d5536836 | |||
| 0d147adfe5 | |||
| 67ba331ba5 | 
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1 +1,2 @@ | |||||||
| build/ | build/ | ||||||
|  | .idea/ | ||||||
							
								
								
									
										109
									
								
								CMakeLists.txt
									
									
									
									
									
								
							
							
						
						
									
										109
									
								
								CMakeLists.txt
									
									
									
									
									
								
							| @ -1,55 +1,118 @@ | |||||||
| cmake_minimum_required(VERSION 3.16) | cmake_minimum_required(VERSION 3.16) | ||||||
| project(CodingGame LANGUAGES C CXX) | project(CodingGame LANGUAGES C CXX) | ||||||
|  |  | ||||||
| # --- deps via vcpkg --- | # ---------- Build-type defaults (only affects single-config generators like Ninja/Make) ---------- | ||||||
| # (vcpkg installs decide static vs shared; no "SDL3-shared" component needed) | if(NOT CMAKE_CONFIGURATION_TYPES AND NOT CMAKE_BUILD_TYPE) | ||||||
| find_package(SDL3 CONFIG REQUIRED) |   set(CMAKE_BUILD_TYPE Release CACHE STRING "Choose the type of build." FORCE) | ||||||
| find_package(OpenGL REQUIRED) | endif() | ||||||
| find_package(GLEW CONFIG REQUIRED) |  | ||||||
| find_package(glm CONFIG REQUIRED) | # ---------- Dependencies ---------- | ||||||
|  | if (UNIX) | ||||||
|  |   include(FetchContent) | ||||||
|  |   FetchContent_Declare( | ||||||
|  |     glm | ||||||
|  |     GIT_REPOSITORY https://github.com/g-truc/glm.git | ||||||
|  |     GIT_TAG        bf71a834948186f4097caa076cd2663c69a10e1e # refs/tags/1.0.1 | ||||||
|  |   ) | ||||||
|  |   FetchContent_MakeAvailable(glm) | ||||||
|  |  | ||||||
|  |   find_package(SDL3 REQUIRED CONFIG REQUIRED COMPONENTS SDL3-shared) | ||||||
|  |   find_package(OpenGL REQUIRED) | ||||||
|  |   find_package(GLEW REQUIRED) | ||||||
|  | elseif (MSVC) # vcpkg | ||||||
|  |   find_package(SDL3 CONFIG REQUIRED) | ||||||
|  |   find_package(OpenGL REQUIRED) | ||||||
|  |   find_package(GLEW CONFIG REQUIRED) | ||||||
|  |   find_package(glm CONFIG REQUIRED) | ||||||
|  | endif() | ||||||
|  |  | ||||||
| # --- exe --- |  | ||||||
| add_executable(CodingGame | add_executable(CodingGame | ||||||
|   src/prelude.cpp |   src/IO/parser.cpp | ||||||
|   src/file_manager.cpp |   src/IO/file_manager.cpp | ||||||
|   src/shader.cpp |  | ||||||
|   src/block.cpp |   src/renderer/debug.cpp | ||||||
|   src/vertex.cpp |   src/renderer/basics.cpp | ||||||
|   src/texture.cpp |   src/renderer/mesh.cpp | ||||||
|   src/model.cpp |   src/renderer/shader.cpp | ||||||
|  |   src/renderer/texture.cpp | ||||||
|  |   src/renderer/wavefront.cpp | ||||||
|  |  | ||||||
|   src/main.cpp |   src/main.cpp | ||||||
| ) | ) | ||||||
|  |  | ||||||
| set_property(TARGET CodingGame PROPERTY CXX_STANDARD 17) | set_property(TARGET CodingGame PROPERTY CXX_STANDARD 17) | ||||||
| set_property(TARGET CodingGame PROPERTY CXX_STANDARD_REQUIRED ON) | set_property(TARGET CodingGame PROPERTY CXX_STANDARD_REQUIRED ON) | ||||||
|  |  | ||||||
| file(COPY ${CMAKE_SOURCE_DIR}/src/shaders DESTINATION ${CMAKE_BINARY_DIR}/) | file(COPY src/shaders DESTINATION ${CMAKE_BINARY_DIR}/) | ||||||
|  |  | ||||||
| target_include_directories(CodingGame PRIVATE | target_include_directories(CodingGame PRIVATE | ||||||
|   ${CMAKE_SOURCE_DIR}/include |   ${CMAKE_SOURCE_DIR}/include | ||||||
|   ${CMAKE_SOURCE_DIR}/contrib |   ${CMAKE_SOURCE_DIR}/contrib | ||||||
| ) | ) | ||||||
|  |  | ||||||
| target_link_libraries(CodingGame PRIVATE | target_link_libraries(CodingGame PRIVATE | ||||||
|   glm::glm |   SDL3::SDL3 | ||||||
|   OpenGL::GL |   OpenGL::GL | ||||||
|   SDL3::SDL3           # vcpkg’s SDL3 target |  | ||||||
|   GLEW::GLEW |   GLEW::GLEW | ||||||
|  |   glm::glm | ||||||
| ) | ) | ||||||
|  |  | ||||||
| # Debug flags per toolchain | # ---------- Visibility (helps optimizer & smaller binaries on Release) ---------- | ||||||
|  | # Only affects non-Windows compilers | ||||||
|  | set_target_properties(CodingGame PROPERTIES | ||||||
|  |   CXX_VISIBILITY_PRESET hidden | ||||||
|  |   VISIBILITY_INLINES_HIDDEN YES | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | # ---------- Per-config flags ---------- | ||||||
|  | # Debug flags (your original intent, kept) | ||||||
| if (MSVC) | if (MSVC) | ||||||
|   target_compile_options(CodingGame PRIVATE $<$<CONFIG:Debug>:/Zi>) |   target_compile_options(CodingGame PRIVATE $<$<CONFIG:Debug>:/Zi>) | ||||||
|   target_link_options(CodingGame PRIVATE $<$<CONFIG:Debug>:/DEBUG:FULL>) |   target_link_options(CodingGame    PRIVATE $<$<CONFIG:Debug>:/DEBUG:FULL>) | ||||||
| else() | else() | ||||||
|   target_compile_options(CodingGame PRIVATE $<$<CONFIG:Debug>:-ggdb>) |   target_compile_options(CodingGame PRIVATE $<$<CONFIG:Debug>:-ggdb>) | ||||||
| endif() | endif() | ||||||
|  |  | ||||||
| # --- copy runtime DLLs next to the exe on Windows --- | # Release flags | ||||||
| # (CMake 3.21+) | if (MSVC) | ||||||
|  |   # /O2: optimize speed, /GL: whole program opt (LTCG), /DNDEBUG: disable asserts | ||||||
|  |   target_compile_options(CodingGame PRIVATE | ||||||
|  |     $<$<CONFIG:Release>:/O2> | ||||||
|  |     $<$<CONFIG:Release>:/DNDEBUG> | ||||||
|  |     $<$<CONFIG:RelWithDebInfo>:/O2> | ||||||
|  |   ) | ||||||
|  |   # Link-time codegen & extra linker opts for smaller/faster binaries | ||||||
|  |   target_link_options(CodingGame PRIVATE | ||||||
|  |     $<$<CONFIG:Release>:/LTCG /OPT:ICF /OPT:REF> | ||||||
|  |     $<$<CONFIG:RelWithDebInfo>:/LTCG /OPT:ICF /OPT:REF> | ||||||
|  |   ) | ||||||
|  | else() | ||||||
|  |   # GCC/Clang | ||||||
|  |   # -O3 for max opts, -ffast-math optional but can be risky; we keep it OFF by default. | ||||||
|  |   option(CODINGGAME_USE_MARCH_NATIVE "Enable -march=native on Release for this machine" ON) | ||||||
|  |   target_compile_options(CodingGame PRIVATE | ||||||
|  |     $<$<CONFIG:Release>:-O3> | ||||||
|  |     $<$<AND:$<CONFIG:Release>,$<BOOL:${CODINGGAME_USE_MARCH_NATIVE}>>:-march=native> | ||||||
|  |     $<$<CONFIG:Release>:-DNDEBUG> | ||||||
|  |     $<$<CONFIG:RelWithDebInfo>:-O3 -g> | ||||||
|  |   ) | ||||||
|  |   # Linker: enable LTO when available; optionally strip symbols on non-Apple | ||||||
|  |   include(CheckIPOSupported)  # IPO == LTO in CMake terms | ||||||
|  |   check_ipo_supported(RESULT ipo_supported OUTPUT ipo_msg) | ||||||
|  |   if(ipo_supported) | ||||||
|  |     set_property(TARGET CodingGame PROPERTY INTERPROCEDURAL_OPTIMIZATION_RELEASE TRUE) | ||||||
|  |     set_property(TARGET CodingGame PROPERTY INTERPROCEDURAL_OPTIMIZATION_RELWITHDEBINFO TRUE) | ||||||
|  |   endif() | ||||||
|  |  | ||||||
|  |   if (CMAKE_CXX_COMPILER_ID STREQUAL "GNU" AND NOT APPLE) | ||||||
|  |     # -s strips symbols at link stage (use only for pure Release) | ||||||
|  |     target_link_options(CodingGame PRIVATE $<$<CONFIG:Release>:-s>) | ||||||
|  |   endif() | ||||||
|  | endif() | ||||||
|  |  | ||||||
|  | # ---------- Windows: copy runtime DLLs ---------- | ||||||
| if (WIN32) | if (WIN32) | ||||||
|   add_custom_command(TARGET CodingGame POST_BUILD |   add_custom_command(TARGET CodingGame POST_BUILD | ||||||
|     COMMAND ${CMAKE_COMMAND} -E copy_if_different |     COMMAND ${CMAKE_COMMAND} -E copy_if_different | ||||||
|       $<TARGET_RUNTIME_DLLS:CodingGame> $<TARGET_FILE_DIR:CodingGame> |       $<TARGET_RUNTIME_DLLS:CodingGame> $<TARGET_FILE_DIR:CodingGame> | ||||||
|     COMMAND_EXPAND_LISTS) |     COMMAND_EXPAND_LISTS) | ||||||
| endif() | endif() | ||||||
							
								
								
									
										52
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,52 @@ | |||||||
|  |  | ||||||
|  |  | ||||||
|  | # Project Description | ||||||
|  |  | ||||||
|  | This is a basic future game engine for OpenGL 3D rendered games | ||||||
|  |  | ||||||
|  | ## Building on Windows | ||||||
|  |  | ||||||
|  | In order to configure and run project on windows platform accomplish several steps. | ||||||
|  |  | ||||||
|  | ### Configuring | ||||||
|  |  | ||||||
|  | ```console | ||||||
|  | cmake -S . -B build -G "Visual Studio 17 2022" -A x64 -DCMAKE_TOOLCHAIN_FILE=C:/vcpkg/scripts/buildsystems/vcpkg.cmake -DVCPKG_TARGET_TRIPLET=x64-windows -DCMAKE_BUILD_TYPE=Release | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Building | ||||||
|  |  | ||||||
|  | ```console | ||||||
|  | cmake --build build --config Release | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Static Linking | ||||||
|  |  | ||||||
|  | For static linking you just need to modify the configure command as follows: | ||||||
|  |  | ||||||
|  | ```console | ||||||
|  | cmake -S . -B build -G "Visual Studio 17 2022" -A x64 -DCMAKE_TOOLCHAIN_FILE=C:/vcpkg/scripts/buildsystems/vcpkg.cmake -DVCPKG_TARGET_TRIPLET=x64-windows-static -DCMAKE_BUILD_TYPE=Release | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Multi-GPU Devices | ||||||
|  |  | ||||||
|  | If you want to use non-primary GPU on your device when launching the game specifically on Linux you should specify additional environment variables before running. For example in my case I have a hybrid gaming laptop with 2 GPUs AMD from CPU and NVIDIA discrete. | ||||||
|  |  | ||||||
|  | The run command in that case would look following: | ||||||
|  |  | ||||||
|  | ```console | ||||||
|  | __NV_PRIME_RENDER_OFFLOAD=1 __GLX_VENDOR_LIBRARY_NAME=nvidia ./build/CodingGame | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## TODO List | ||||||
|  |  | ||||||
|  | ### Optimizations | ||||||
|  |  | ||||||
|  | 🚀 Summary of Speedups | ||||||
|  | - Replace toElement / toMtlElement string comparisons with char-based switches. | ||||||
|  | - Replace std::stoi with a custom fast Parser::TakeIndex. | ||||||
|  | - Pre-reserve vectors for vertices, normals, texcoords, meshes. | ||||||
|  | - Load whole file into memory before parsing (fastest for large OBJs). | ||||||
|  | - Defer texture loading until after parsing. | ||||||
|  | - Store material pointers in meshes → no runtime lookups in render. | ||||||
|  | - Inline parsing functions. | ||||||
							
								
								
									
										25
									
								
								WINDOWS.md
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								WINDOWS.md
									
									
									
									
									
								
							| @ -1,25 +0,0 @@ | |||||||
|  |  | ||||||
| # Setup VCPKG |  | ||||||
|  |  | ||||||
| ```console |  | ||||||
| git clone https://github.com/microsoft/vcpkg C:\vcpkg |  | ||||||
| C:\vcpkg\bootstrap-vcpkg.bat |  | ||||||
| # install deps for 64-bit Windows |  | ||||||
| C:\vcpkg\vcpkg install sdl3 glew glm --triplet x64-windows |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| # Configure |  | ||||||
|  |  | ||||||
| ```console |  | ||||||
| cmake -S . -B build ` |  | ||||||
|   -G "Visual Studio 17 2022" -A x64 ` |  | ||||||
|   -DCMAKE_TOOLCHAIN_FILE=C:/vcpkg/scripts/buildsystems/vcpkg.cmake ` |  | ||||||
|   -DVCPKG_TARGET_TRIPLET=x64-windows ` |  | ||||||
|   -DCMAKE_BUILD_TYPE=Debug |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| # Build |  | ||||||
|  |  | ||||||
| ```console |  | ||||||
| cmake --build build --config Debug |  | ||||||
| ``` |  | ||||||
							
								
								
									
										20
									
								
								include/IO/parser.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								include/IO/parser.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,20 @@ | |||||||
|  | #ifndef PARSER_H_ | ||||||
|  | #define PARSER_H_ | ||||||
|  |  | ||||||
|  | // Very fast OBJ/MTL line parser | ||||||
|  | class Parser { | ||||||
|  | private: | ||||||
|  |     char* m_sv; | ||||||
|  | public: | ||||||
|  |     Parser(char* sv) : m_sv(sv) {} | ||||||
|  | public: | ||||||
|  |     void SkipSpaces(); | ||||||
|  |     char* TakeWord(); | ||||||
|  |     float TakeFloat(); | ||||||
|  |     int TakeInt(); | ||||||
|  |     bool TakeFaceIndices(int& vi, int& ti, int& ni); | ||||||
|  |     char* TakeUntil(char d); | ||||||
|  |     int TakeIndex(int baseCount); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #endif // PARSER_H_ | ||||||
| @ -1,16 +0,0 @@ | |||||||
| #ifndef BLOCK_H_ |  | ||||||
| #define BLOCK_H_ |  | ||||||
| #include <glm/glm.hpp> |  | ||||||
|  |  | ||||||
| class Block { |  | ||||||
| private: |  | ||||||
|     glm::vec3 m_position; |  | ||||||
|     glm::vec4 m_color; |  | ||||||
| public: |  | ||||||
|     Block(glm::vec3 position, glm::vec4 color); |  | ||||||
| public: |  | ||||||
|     inline glm::vec3 Position() const { return m_position; } |  | ||||||
|     inline glm::vec4 Color() const { return m_color; } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| #endif |  | ||||||
							
								
								
									
										134
									
								
								include/model.h
									
									
									
									
									
								
							
							
						
						
									
										134
									
								
								include/model.h
									
									
									
									
									
								
							| @ -1,134 +0,0 @@ | |||||||
| #ifndef MODEL_H_ |  | ||||||
| #define MODEL_H_ |  | ||||||
| #include <vector> |  | ||||||
| #include <string> |  | ||||||
| #include <unordered_map> |  | ||||||
| #include <glm/glm.hpp> |  | ||||||
| #include <memory> |  | ||||||
|  |  | ||||||
| #include "shader.h" |  | ||||||
| #include "texture.h" |  | ||||||
|  |  | ||||||
| enum ObjElement { OHASH, MTLLIB, USEMTL, O, V, VN, VT, F, OUNKNOWN }; |  | ||||||
| enum MtlElement { MHASH, NEWMTL, NS, KA, KS, KD, NI, D, ILLUM, MAP_KD, MAP_KA, MUNKNOWN }; |  | ||||||
|  |  | ||||||
| class Vertex { |  | ||||||
| private: |  | ||||||
|     glm::vec3 m_position; |  | ||||||
|     glm::vec3 m_normal; |  | ||||||
|     glm::vec2 m_texCoord; |  | ||||||
| public: |  | ||||||
|     Vertex(glm::vec3 position, glm::vec3 normal, glm::vec2 texCoord) |  | ||||||
|         : m_position(position), m_normal(normal), m_texCoord(texCoord) {} |  | ||||||
| public: |  | ||||||
|     static void DefineAttrib(); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class FaceItem { |  | ||||||
| private: |  | ||||||
|     unsigned int m_vIndex; |  | ||||||
|     unsigned int m_nIndex; |  | ||||||
|     unsigned int m_tIndex; |  | ||||||
| public: |  | ||||||
|     FaceItem() |  | ||||||
|         : m_vIndex(0), m_nIndex(0), m_tIndex(0) {} |  | ||||||
|     FaceItem(unsigned int vI, unsigned int nI, unsigned int tI) |  | ||||||
|         : m_vIndex(vI), m_nIndex(nI), m_tIndex(tI) {} |  | ||||||
| public: |  | ||||||
|     inline const unsigned int GetVertex() const { return m_vIndex; } |  | ||||||
|     inline const unsigned int GetNormal() const { return m_nIndex; } |  | ||||||
|     inline const unsigned int GetTex() const { return m_tIndex; } |  | ||||||
| public: |  | ||||||
|     inline void SetVertex(unsigned int vIndex) { m_vIndex = vIndex; } |  | ||||||
|     inline void SetNorm(unsigned int nIndex) { m_nIndex = nIndex; } |  | ||||||
|     inline void SetTex(unsigned int tIndex) { m_tIndex = tIndex; } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class Face { |  | ||||||
| private: |  | ||||||
|     std::vector<FaceItem> m_items; |  | ||||||
| public: |  | ||||||
|     Face() |  | ||||||
|         : m_items(std::vector<FaceItem>()) {} |  | ||||||
| public: |  | ||||||
|     void PushItem(const FaceItem& item); |  | ||||||
| public: |  | ||||||
|     inline const unsigned int GetSize() const { return m_items.size(); } |  | ||||||
|     inline const FaceItem& GetItem(unsigned int index) const { return m_items[index]; } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class Material { |  | ||||||
| private: |  | ||||||
|     glm::vec3 m_ambient { 0.2f, 0.2f, 0.2f }; |  | ||||||
|     glm::vec3 m_diffuse { 0.8f, 0.8f, 0.8f }; |  | ||||||
|     glm::vec3 m_specular { 1.0f, 1.0f, 1.0f }; |  | ||||||
|     float m_shininess { 32.0f }; |  | ||||||
|  |  | ||||||
|     std::unique_ptr<Texture> m_diffuse_tex { nullptr }; |  | ||||||
| public: |  | ||||||
|     Material() = default; |  | ||||||
|     Material(const Material& other) = default; // copy constructor |  | ||||||
|     Material& operator=(const Material& other) = default; |  | ||||||
| public: |  | ||||||
|     inline const glm::vec3 GetAmbientColor() const { return m_ambient; } |  | ||||||
|     inline const glm::vec3 GetDiffuseColor() const { return m_diffuse; } |  | ||||||
|     inline const glm::vec3 GetSpecularColor() const { return m_specular; } |  | ||||||
|     inline const float GetSpecularWeight() const { return m_shininess; } |  | ||||||
|     inline const bool HasDiffuseTexture() const { return m_diffuse_tex != nullptr; } |  | ||||||
|     inline const Texture* GetDiffuseTexture() const { return m_diffuse_tex.get(); } |  | ||||||
| public: |  | ||||||
|     inline void SetAmbientColor(glm::vec3 ambient) { m_ambient = ambient; } |  | ||||||
|     inline void SetDiffuseColor(glm::vec3 diffuse) { m_diffuse = diffuse; } |  | ||||||
|     inline void SetSpecularColor(glm::vec3 specular) { m_specular = specular; } |  | ||||||
|     inline void SetSpecularWeight(float weight) { m_shininess = weight; } |  | ||||||
|     inline void SetDiffuseTexture(std::unique_ptr<Texture>&& texture) { m_diffuse_tex = std::move(texture); } |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class Mesh { |  | ||||||
| public: // TODO: abstract away |  | ||||||
|     unsigned int m_vao, m_vbo, m_ebo; |  | ||||||
|     std::vector<Vertex> m_vertexBuffer; |  | ||||||
|     std::vector<unsigned int> m_indexBuffer; |  | ||||||
| public: // TODO: abstract away |  | ||||||
|     void Bind() { glBindVertexArray(m_vao); } |  | ||||||
|     void Unbind() { glBindVertexArray(0); } |  | ||||||
|     void Upload(); |  | ||||||
| public: |  | ||||||
|     std::string materialName; |  | ||||||
| public: |  | ||||||
|     Mesh(); |  | ||||||
| public: |  | ||||||
|     void Render(); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class Object { |  | ||||||
| private: |  | ||||||
|     std::string m_name; |  | ||||||
|     std::vector<glm::vec3> m_vertices; |  | ||||||
|     std::vector<glm::vec3> m_normals; |  | ||||||
|     std::vector<glm::vec2> m_texCoords; |  | ||||||
|  |  | ||||||
|     std::vector<Mesh> m_meshes; |  | ||||||
|  |  | ||||||
|     std::unordered_map<std::string, std::shared_ptr<Material>> m_materials; |  | ||||||
| private: |  | ||||||
|     static inline int NormalizeIndex(const std::string &s, int baseCount); |  | ||||||
|  |  | ||||||
| private: |  | ||||||
|     Object(); |  | ||||||
| public: |  | ||||||
|     static Object LoadFile(const std::string& filename); |  | ||||||
|  |  | ||||||
| private: |  | ||||||
|     void LoadMaterials(const std::string& filename); |  | ||||||
| private: |  | ||||||
|     void AddMaterial(std::string name, std::shared_ptr<Material> material); |  | ||||||
|     std::shared_ptr<Material> GetMaterial(std::string name); |  | ||||||
| private: |  | ||||||
|     Mesh& GetLastMesh(); |  | ||||||
|     void CreateNewMesh(const std::string& materialName); |  | ||||||
| public: |  | ||||||
|     void Render(Shader& shader); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| #endif // MODEL_H_ |  | ||||||
| @ -1,39 +0,0 @@ | |||||||
| #ifndef PRELUDE_H_ |  | ||||||
| #define PRELUDE_H_ |  | ||||||
| // #define GLEW_STATIC |  | ||||||
| #include <GL/glew.h> |  | ||||||
| #include "SDL3/SDL.h" |  | ||||||
|  |  | ||||||
| struct RenderContext { |  | ||||||
|     Uint64 time; |  | ||||||
|     Uint64 prev_time; |  | ||||||
|     bool program_failed = false; |  | ||||||
|     GLuint program = 0; |  | ||||||
|     GLint resolution_location = 0; |  | ||||||
|     GLint time_location = 0; |  | ||||||
|     bool pause = false; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| bool compile_shader_source(const GLchar *source, GLenum shader_type, GLuint *shader); |  | ||||||
|  |  | ||||||
| bool compile_shader_file(const char *file_path, GLenum shader_type, GLuint *shader); |  | ||||||
|  |  | ||||||
| bool link_program(GLuint vert_shader, GLuint frag_shader, GLuint *program); |  | ||||||
|  |  | ||||||
| void reload_shaders(RenderContext*context); |  | ||||||
|  |  | ||||||
| // void key_callback(SDL_Window* window, int key, int scancode, int action, int mods); |  | ||||||
|  |  | ||||||
| void window_size_callback(SDL_Window* window, int width, int height); |  | ||||||
|  |  | ||||||
| void MessageCallback(GLenum source, |  | ||||||
|                      GLenum type, |  | ||||||
|                      GLuint id, |  | ||||||
|                      GLenum severity, |  | ||||||
|                      GLsizei length, |  | ||||||
|                      const GLchar* message, |  | ||||||
|                      const void* userParam); |  | ||||||
|  |  | ||||||
| void process_prelude(RenderContext *context); |  | ||||||
|  |  | ||||||
| #endif // PRELUDE_H_ |  | ||||||
							
								
								
									
										18
									
								
								include/renderer/basics.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								include/renderer/basics.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | |||||||
|  | #ifndef RENDERER_BASICS_H | ||||||
|  | #define RENDERER_BASICS_H | ||||||
|  |  | ||||||
|  | #include <glm/glm.hpp> | ||||||
|  |  | ||||||
|  | class Vertex { | ||||||
|  | private: | ||||||
|  |     glm::vec3 m_position; | ||||||
|  |     glm::vec3 m_normal; | ||||||
|  |     glm::vec2 m_texCoord; | ||||||
|  | public: | ||||||
|  |     Vertex(glm::vec3 position, glm::vec3 normal, glm::vec2 texCoord) | ||||||
|  |         : m_position(position), m_normal(normal), m_texCoord(texCoord) {} | ||||||
|  | public: | ||||||
|  |     static void DefineAttrib(); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #endif // RENDERER_BASICS_H | ||||||
							
								
								
									
										16
									
								
								include/renderer/debug.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								include/renderer/debug.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,16 @@ | |||||||
|  | #ifndef RENDERER_DEBUG_ | ||||||
|  | #define RENDERER_DEBUG_ | ||||||
|  |  | ||||||
|  | #include <iostream> | ||||||
|  |  | ||||||
|  | #include <GL/glew.h> | ||||||
|  |  | ||||||
|  | void MessageCallback(GLenum source, | ||||||
|  |                      GLenum type, | ||||||
|  |                      GLuint id, | ||||||
|  |                      GLenum severity, | ||||||
|  |                      GLsizei length, | ||||||
|  |                      const GLchar* message, | ||||||
|  |                      const void* userParam); | ||||||
|  |  | ||||||
|  | #endif // RENDERER_DEBUG_ | ||||||
							
								
								
									
										42
									
								
								include/renderer/material.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								include/renderer/material.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | #ifndef MATERIAL_H_ | ||||||
|  | #define MATERIAL_H_ | ||||||
|  |  | ||||||
|  | #include <glm/glm.hpp> | ||||||
|  | #include <memory> | ||||||
|  |  | ||||||
|  | #include "texture.h" | ||||||
|  |  | ||||||
|  | class Material { | ||||||
|  | private: | ||||||
|  |     glm::vec3 m_ambient { 0.2f, 0.2f, 0.2f }; | ||||||
|  |     glm::vec3 m_diffuse { 0.8f, 0.8f, 0.8f }; | ||||||
|  |     glm::vec3 m_specular { 1.0f, 1.0f, 1.0f }; | ||||||
|  |     float m_shininess { 32.0f }; | ||||||
|  |     float m_opacity { 1.0f }; | ||||||
|  |     int m_illum { 2 }; | ||||||
|  |  | ||||||
|  |     std::unique_ptr<Texture> m_diffuse_tex { nullptr }; | ||||||
|  | public: | ||||||
|  |     Material() = default; | ||||||
|  |     Material(const Material& other) = default; // copy constructor | ||||||
|  |     Material& operator=(const Material& other) = default; | ||||||
|  | public: | ||||||
|  |     inline const glm::vec3 GetAmbientColor() const { return m_ambient; } | ||||||
|  |     inline const glm::vec3 GetDiffuseColor() const { return m_diffuse; } | ||||||
|  |     inline const glm::vec3 GetSpecularColor() const { return m_specular; } | ||||||
|  |     inline const float GetSpecularWeight() const { return m_shininess; } | ||||||
|  |     inline const bool HasDiffuseTexture() const { return m_diffuse_tex != nullptr; } | ||||||
|  |     inline const Texture* GetDiffuseTexture() const { return m_diffuse_tex.get(); } | ||||||
|  |     inline const float GetOpacity() const { return m_opacity; } | ||||||
|  |     inline const int GetIllumination() const { return m_illum; } | ||||||
|  | public: | ||||||
|  |     inline void SetAmbientColor(glm::vec3 ambient) { m_ambient = ambient; } | ||||||
|  |     inline void SetDiffuseColor(glm::vec3 diffuse) { m_diffuse = diffuse; } | ||||||
|  |     inline void SetSpecularColor(glm::vec3 specular) { m_specular = specular; } | ||||||
|  |     inline void SetSpecularWeight(float weight) { m_shininess = weight; } | ||||||
|  |     inline void SetDiffuseTexture(std::unique_ptr<Texture>&& texture) { m_diffuse_tex = std::move(texture); } | ||||||
|  |     inline void SetOpacity(float opacity) { m_opacity = opacity; } | ||||||
|  |     inline void SetIllumination(float illum) { m_illum = illum; } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #endif // MATERIAL_H_ | ||||||
							
								
								
									
										27
									
								
								include/renderer/mesh.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								include/renderer/mesh.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | |||||||
|  | #ifndef MESH_H_ | ||||||
|  | #define MESH_H_ | ||||||
|  |  | ||||||
|  | #include <vector> | ||||||
|  | #include <string> | ||||||
|  | #include <GL/glew.h> | ||||||
|  |  | ||||||
|  | #include "renderer/basics.h" | ||||||
|  |  | ||||||
|  | class Mesh { | ||||||
|  | public: // TODO: abstract away | ||||||
|  |     unsigned int m_vao, m_vbo, m_ebo; | ||||||
|  |     std::vector<Vertex> m_vertexBuffer; | ||||||
|  |     std::vector<unsigned int> m_indexBuffer; | ||||||
|  | public: // TODO: abstract away | ||||||
|  |     void Bind() { glBindVertexArray(m_vao); } | ||||||
|  |     void Unbind() { glBindVertexArray(0); } | ||||||
|  |     void Upload(); | ||||||
|  | public: | ||||||
|  |     std::string materialName; | ||||||
|  | public: | ||||||
|  |     Mesh(); | ||||||
|  | public: | ||||||
|  |     void Render(); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #endif // MESH_H_ | ||||||
							
								
								
									
										49
									
								
								include/renderer/wavefront.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								include/renderer/wavefront.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,49 @@ | |||||||
|  | #ifndef MODEL_H_ | ||||||
|  | #define MODEL_H_ | ||||||
|  | #include <vector> | ||||||
|  | #include <string> | ||||||
|  | #include <filesystem> | ||||||
|  | #include <unordered_map> | ||||||
|  | #include <glm/glm.hpp> | ||||||
|  | #include <memory> | ||||||
|  |  | ||||||
|  | #include "shader.h" | ||||||
|  | #include "texture.h" | ||||||
|  | #include "renderer/material.h" | ||||||
|  | #include "renderer/basics.h" | ||||||
|  | #include "renderer/mesh.h" | ||||||
|  |  | ||||||
|  | enum ObjElement { OHASH, MTLLIB, USEMTL, O, V, VN, VT, F, OUNKNOWN }; | ||||||
|  | enum MtlElement { MHASH, NEWMTL, NS, KA, KS, KD, NI, D, ILLUM, MAP_KD, MAP_KA, MUNKNOWN }; | ||||||
|  |  | ||||||
|  | class Object { | ||||||
|  | private: | ||||||
|  |     std::string m_name; | ||||||
|  |     std::vector<glm::vec3> m_vertices; | ||||||
|  |     std::vector<glm::vec3> m_normals; | ||||||
|  |     std::vector<glm::vec2> m_texCoords; | ||||||
|  |  | ||||||
|  |     std::vector<Mesh> m_meshes; | ||||||
|  |  | ||||||
|  |     std::unordered_map<std::string, std::shared_ptr<Material>> m_materials; | ||||||
|  | private: | ||||||
|  |     static inline int NormalizeIndex(int idx, int baseCount); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     Object(); | ||||||
|  | public: | ||||||
|  |     static Object LoadFile(const std::string& filename); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     void LoadMaterials(const std::filesystem::path& filename); | ||||||
|  | private: | ||||||
|  |     void AddMaterial(std::string name, std::shared_ptr<Material> material); | ||||||
|  |     std::shared_ptr<Material> GetMaterial(std::string name); | ||||||
|  | private: | ||||||
|  |     Mesh& GetLastMesh(); | ||||||
|  |     void CreateNewMesh(const std::string& materialName); | ||||||
|  | public: | ||||||
|  |     void Render(Shader& shader); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #endif // MODEL_H_ | ||||||
| @ -1,37 +0,0 @@ | |||||||
| #ifndef VERTEX_H_ |  | ||||||
| #define VERTEX_H_ |  | ||||||
| #include <vector> |  | ||||||
| #include <glm/glm.hpp> |  | ||||||
|  |  | ||||||
| #include "block.h" |  | ||||||
|  |  | ||||||
| class Point { |  | ||||||
| private: |  | ||||||
|     glm::vec3 m_position; |  | ||||||
|     glm::vec3 m_normal; |  | ||||||
|     glm::vec4 m_color; |  | ||||||
| public: |  | ||||||
|     Point(glm::vec3 position, glm::vec3 normal, glm::vec4 color); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class Vertices { |  | ||||||
| private: |  | ||||||
|     std::vector<Point> m_items; |  | ||||||
|     std::vector<unsigned int> m_indices; |  | ||||||
|  |  | ||||||
|     unsigned int m_vao; |  | ||||||
|     unsigned int m_vbo; |  | ||||||
|     unsigned int m_ebo; |  | ||||||
| public: |  | ||||||
|     Vertices(); |  | ||||||
| public: // GPU |  | ||||||
|     void Bind(); |  | ||||||
|     void Unbind(); |  | ||||||
|     void Draw(); |  | ||||||
|     void Upload(); |  | ||||||
| public: |  | ||||||
|     void PushVertex(const Point& point); |  | ||||||
|     void PushIndex(unsigned int index); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| #endif // VERTEX_H_ |  | ||||||
| @ -1,4 +1,4 @@ | |||||||
| #include "file_manager.h" | #include "IO/file_manager.h" | ||||||
| 
 | 
 | ||||||
| #include <fstream> | #include <fstream> | ||||||
| #include <iostream> | #include <iostream> | ||||||
							
								
								
									
										123
									
								
								src/IO/parser.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								src/IO/parser.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,123 @@ | |||||||
|  | #include <charconv>   // for std::from_chars (C++17+) | ||||||
|  | #include <cstdlib>    // for strtof (fallback) | ||||||
|  | #include <cstring> | ||||||
|  |  | ||||||
|  | #include "IO/parser.h" | ||||||
|  |  | ||||||
|  | // Skip whitespace | ||||||
|  | void Parser::SkipSpaces() { | ||||||
|  |     while (*m_sv == ' ' || *m_sv == '\t') ++m_sv; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int Parser::TakeIndex(int baseCount) { | ||||||
|  |     if (!m_sv || *m_sv == '\0') return -1; | ||||||
|  |  | ||||||
|  |     bool neg = (*m_sv == '-'); | ||||||
|  |     if (neg) ++m_sv; | ||||||
|  |  | ||||||
|  |     int idx = 0; | ||||||
|  |     while (*m_sv >= '0' && *m_sv <= '9') { | ||||||
|  |         idx = idx * 10 + (*m_sv - '0'); | ||||||
|  |         ++m_sv; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (neg) return baseCount + (-idx); | ||||||
|  |     return idx > 0 ? idx - 1 : -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Get next whitespace-delimited word | ||||||
|  | char* Parser::TakeWord() { | ||||||
|  |     SkipSpaces(); | ||||||
|  |     if (*m_sv == '\0' || *m_sv == '\n' || *m_sv == '\r') return nullptr; | ||||||
|  |  | ||||||
|  |     char* start = m_sv; | ||||||
|  |     while (*m_sv && *m_sv != ' ' && *m_sv != '\t' && *m_sv != '\n' && *m_sv != '\r') | ||||||
|  |         ++m_sv; | ||||||
|  |  | ||||||
|  |     if (*m_sv) { *m_sv = '\0'; ++m_sv; } | ||||||
|  |     return start; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Parse a float quickly | ||||||
|  | float Parser::TakeFloat() { | ||||||
|  |     SkipSpaces(); | ||||||
|  |     if (*m_sv == '\0') return 0.0f; | ||||||
|  |  | ||||||
|  | #if __cpp_lib_to_chars >= 201611L | ||||||
|  |     float value = 0.0f; | ||||||
|  |     auto result = std::from_chars(m_sv, m_sv + std::strlen(m_sv), value); | ||||||
|  |     m_sv = const_cast<char*>(result.ptr); | ||||||
|  |     return value; | ||||||
|  | #else | ||||||
|  |     char* end; | ||||||
|  |     float value = std::strtof(m_sv, &end); | ||||||
|  |     m_sv = end; | ||||||
|  |     return value; | ||||||
|  | #endif | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Parse an integer quickly | ||||||
|  | int Parser::TakeInt() { | ||||||
|  |     SkipSpaces(); | ||||||
|  |     if (*m_sv == '\0') return 0; | ||||||
|  |  | ||||||
|  | #if __cpp_lib_to_chars >= 201611L | ||||||
|  |     int value = 0; | ||||||
|  |     auto result = std::from_chars(m_sv, m_sv + std::strlen(m_sv), value); | ||||||
|  |     m_sv = const_cast<char*>(result.ptr); | ||||||
|  |     return value; | ||||||
|  | #else | ||||||
|  |     char* end; | ||||||
|  |     int value = static_cast<int>(std::strtol(m_sv, &end, 10)); | ||||||
|  |     m_sv = end; | ||||||
|  |     return value; | ||||||
|  | #endif | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Take everything until delimiter (mutates buffer) | ||||||
|  | char* Parser::TakeUntil(char d) { | ||||||
|  |     SkipSpaces(); | ||||||
|  |     if (*m_sv == '\0') return nullptr; | ||||||
|  |  | ||||||
|  |     char* start = m_sv; | ||||||
|  |     while (*m_sv && *m_sv != d && *m_sv != '\n' && *m_sv != '\r') | ||||||
|  |         ++m_sv; | ||||||
|  |  | ||||||
|  |     if (*m_sv) { *m_sv = '\0'; ++m_sv; } | ||||||
|  |     return start; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Parser.h (or Parser.cpp) | ||||||
|  | // Parse one face element at current position. | ||||||
|  | // Accepts formats: "v", "v/t", "v//n", "v/t/n" | ||||||
|  | // Returns true if a token was parsed, false if no more tokens on the line. | ||||||
|  | bool Parser::TakeFaceIndices(int &vi, int &ti, int &ni) { | ||||||
|  |     SkipSpaces(); | ||||||
|  |     if (*m_sv == '\0' || *m_sv == '\n' || *m_sv == '\r') { | ||||||
|  |         vi = ti = ni = 0; // sentinel raw value meaning "no token" | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // parse vertex index (may be negative) | ||||||
|  |     vi = static_cast<int>(std::strtol(m_sv, &m_sv, 10)); | ||||||
|  |  | ||||||
|  |     ti = ni = 0; // 0 = not present (raw) | ||||||
|  |     if (*m_sv == '/') { | ||||||
|  |         ++m_sv; // skip '/' | ||||||
|  |         // texcoord index (optional) | ||||||
|  |         if (*m_sv != '/' && *m_sv != ' ' && *m_sv != '\0' && *m_sv != '\n' && *m_sv != '\r') { | ||||||
|  |             ti = static_cast<int>(std::strtol(m_sv, &m_sv, 10)); | ||||||
|  |         } | ||||||
|  |         if (*m_sv == '/') { | ||||||
|  |             ++m_sv; // skip second '/' | ||||||
|  |             // normal index (optional) | ||||||
|  |             if (*m_sv != ' ' && *m_sv != '\0' && *m_sv != '\n' && *m_sv != '\r') { | ||||||
|  |                 ni = static_cast<int>(std::strtol(m_sv, &m_sv, 10)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // At this point m_sv is either at whitespace, end, or next token char. | ||||||
|  |     // Do NOT mutate indices (leave them raw). Let NormalizeIndex handle conversion. | ||||||
|  |     return true; | ||||||
|  | } | ||||||
| @ -1,6 +0,0 @@ | |||||||
| #include "block.h" |  | ||||||
|  |  | ||||||
| Block::Block(glm::vec3 position, glm::vec4 color) { |  | ||||||
|     this->m_position = position; |  | ||||||
|     this->m_color = m_color; |  | ||||||
| } |  | ||||||
							
								
								
									
										229
									
								
								src/main.bkp.cpp
									
									
									
									
									
								
							
							
						
						
									
										229
									
								
								src/main.bkp.cpp
									
									
									
									
									
								
							| @ -1,229 +0,0 @@ | |||||||
| #include <iostream> |  | ||||||
| #include <vector> |  | ||||||
| #include <glm/glm.hpp> |  | ||||||
| #include <glm/ext/matrix_transform.hpp> |  | ||||||
| #include <glm/ext/matrix_clip_space.hpp> |  | ||||||
| #include <GL/glew.h> |  | ||||||
| #include <SDL3/SDL.h> |  | ||||||
|  |  | ||||||
| #include "shader.h" |  | ||||||
| #include "file_manager.h" |  | ||||||
| #include "prelude.h" |  | ||||||
| #include "block.h" |  | ||||||
| #include "vertex.h" |  | ||||||
| #include "model.h" |  | ||||||
|  |  | ||||||
| #define WIDTH 1024 |  | ||||||
| #define HEIGHT 768 |  | ||||||
|  |  | ||||||
| int main() { |  | ||||||
|     SDL_Init(SDL_INIT_VIDEO | SDL_INIT_EVENTS); |  | ||||||
|  |  | ||||||
|     SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 4); |  | ||||||
|     SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 6); |  | ||||||
|     SDL_GL_SetAttribute(SDL_GL_CONTEXT_PROFILE_MASK, SDL_GL_CONTEXT_PROFILE_CORE); |  | ||||||
|  |  | ||||||
|     SDL_GL_SetAttribute(SDL_GL_MULTISAMPLEBUFFERS, 1); |  | ||||||
|     SDL_GL_SetAttribute(SDL_GL_MULTISAMPLESAMPLES, 8); |  | ||||||
|  |  | ||||||
|     SDL_Window *window = SDL_CreateWindow("OpenGL Test", WIDTH, HEIGHT, SDL_WINDOW_OPENGL|SDL_WINDOW_RESIZABLE|SDL_WINDOW_ALWAYS_ON_TOP); |  | ||||||
|  |  | ||||||
|     SDL_GLContext glcontext = SDL_GL_CreateContext(window); |  | ||||||
|  |  | ||||||
|     glewExperimental = GL_TRUE; |  | ||||||
|     if (GLEW_OK != glewInit()) { |  | ||||||
|         fprintf(stderr, "Could not initialize GLEW!\n"); |  | ||||||
|         SDL_GL_DestroyContext(glcontext); |  | ||||||
|         SDL_DestroyWindow(window); |  | ||||||
|         exit(1); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     glEnable(GL_DEBUG_OUTPUT); |  | ||||||
|     glEnable(GL_DEPTH_TEST); |  | ||||||
|     glDebugMessageCallback(MessageCallback, 0); |  | ||||||
|  |  | ||||||
|     // brightness multipliers for faces |  | ||||||
|     const float FACE_BRIGHTNESS[6] = { |  | ||||||
|         1.0f,  // front |  | ||||||
|         0.7f,  // right |  | ||||||
|         0.5f,  // back |  | ||||||
|         0.7f,  // left |  | ||||||
|         1.2f,  // top |  | ||||||
|         0.4f   // bottom |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     // position, normal, color |  | ||||||
|     glm::vec4 cubeColor = {1.0f, 0.5f, 0.31f, 1.0f}; |  | ||||||
|     std::vector<Point> cubeVerts = { |  | ||||||
|         // front face (z = 0, normal = +Z) |  | ||||||
|         { {-0.5f, -0.5f, 0.5f}, {0.f, 0.f, 1.f}, cubeColor }, |  | ||||||
|         { {0.5f, -0.5f, 0.5f}, {0.f, 0.f, 1.f}, cubeColor }, |  | ||||||
|         { {0.5f, 0.5f, 0.5f}, {0.f, 0.f, 1.f}, cubeColor }, |  | ||||||
|         { {-0.5f, 0.5f, 0.5f}, {0.f, 0.f, 1.f}, cubeColor }, |  | ||||||
|  |  | ||||||
|         // back face (z = 1, normal = -Z) |  | ||||||
|         { {-0.5f, -0.5f, -0.5f}, {0.f, 0.f, -1.f}, cubeColor }, |  | ||||||
|         { {0.5f, -0.5f, -0.5f}, {0.f, 0.f, -1.f}, cubeColor }, |  | ||||||
|         { {0.5f, 0.5f, -0.5f}, {0.f, 0.f, -1.f}, cubeColor }, |  | ||||||
|         { {-0.5f, 0.5f, -0.5f}, {0.f, 0.f, -1.f}, cubeColor }, |  | ||||||
|  |  | ||||||
|         // left face (x = 0, normal = -X) |  | ||||||
|         { {-0.5f, -0.5f, -0.5f}, {-1.f, 0.f, 0.f}, cubeColor }, |  | ||||||
|         { {-0.5f, -0.5f, 0.5f}, {-1.f, 0.f, 0.f}, cubeColor }, |  | ||||||
|         { {-0.5f, 0.5f, 0.5f}, {-1.f, 0.f, 0.f}, cubeColor }, |  | ||||||
|         { {-0.5f, 0.5f, -0.5f}, {-1.f, 0.f, 0.f}, cubeColor }, |  | ||||||
|  |  | ||||||
|         // right face (x = 1, normal = +X) |  | ||||||
|         { {0.5f, -0.5f, -0.5f}, {1.f, 0.f, 0.f}, cubeColor }, |  | ||||||
|         { {0.5f, -0.5f, 0.5f}, {1.f, 0.f, 0.f}, cubeColor }, |  | ||||||
|         { {0.5f, 0.5f, 0.5f}, {1.f, 0.f, 0.f}, cubeColor }, |  | ||||||
|         { {0.5f, 0.5f, -0.5f}, {1.f, 0.f, 0.f}, cubeColor }, |  | ||||||
|  |  | ||||||
|         // top face (y = 1, normal = +Y) |  | ||||||
|         { {-0.5f, 0.5f, 0.5f}, {0.f, 1.f, 0.f}, cubeColor }, |  | ||||||
|         { {0.5f, 0.5f, 0.5f}, {0.f, 1.f, 0.f}, cubeColor }, |  | ||||||
|         { {0.5f, 0.5f, -0.5f}, {0.f, 1.f, 0.f}, cubeColor }, |  | ||||||
|         { {-0.5f, 0.5f, -0.5f}, {0.f, 1.f, 0.f}, cubeColor }, |  | ||||||
|  |  | ||||||
|         // bottom face (y = 0, normal = -Y) |  | ||||||
|         { {-0.5f, -0.5f, 0.5f}, {0.f, -1.f, 0.f}, cubeColor }, |  | ||||||
|         { {0.5f, -0.5f, 0.5f}, {0.f, -1.f, 0.f}, cubeColor }, |  | ||||||
|         { {0.5f, -0.5f, -0.5f}, {0.f, -1.f, 0.f}, cubeColor }, |  | ||||||
|         { {-0.5f, -0.5f, -0.5f}, {0.f, -1.f, 0.f}, cubeColor }, |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     std::vector<unsigned int> cubeIndices = { |  | ||||||
|         0,1,2,  2,3,0,      // front |  | ||||||
|         4,5,6,  6,7,4,      // back |  | ||||||
|         8,9,10, 10,11,8,    // left |  | ||||||
|         12,13,14, 14,15,12, // right |  | ||||||
|         16,17,18, 18,19,16, // top |  | ||||||
|         20,21,22, 22,23,20  // bottom |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     Vertices vertices; |  | ||||||
|     for (auto &v : cubeVerts) vertices.PushVertex(v); |  | ||||||
|     for (auto i : cubeIndices) vertices.PushIndex(i); |  | ||||||
|     vertices.Upload(); |  | ||||||
|  |  | ||||||
|     Shader simpleShader; |  | ||||||
|     simpleShader.init( |  | ||||||
|         FileManager::read("./src/shaders/simple.vs"), |  | ||||||
|         FileManager::read("./src/shaders/simple.fs") |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     int screenWidth = WIDTH, screenHeight = HEIGHT; |  | ||||||
|  |  | ||||||
|     glm::vec3 cameraPosition(0.f, 0.f, 2.f); |  | ||||||
|     glm::vec3 cameraViewDirection(0.f, 0.f, -1.f); |  | ||||||
|     // glm::vec3 lightPosition(1.f, 3.5f, -2.f); |  | ||||||
|     glm::vec3 lightPosition = cameraPosition; |  | ||||||
|  |  | ||||||
|     glm::mat4 model(1.f); |  | ||||||
|  |  | ||||||
|     glm::mat4 view = glm::lookAt( |  | ||||||
|         cameraPosition, |  | ||||||
|         cameraPosition + cameraViewDirection, |  | ||||||
|         glm::vec3(0.f, 1.f, 0.f) |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     glm::mat4 projection = glm::perspective( |  | ||||||
|         (float)M_PI_2, |  | ||||||
|         (float)screenWidth / (float)screenHeight, |  | ||||||
|         0.01f, |  | ||||||
|         100.0f |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     float angle = 3.45f; |  | ||||||
|     Uint64 lastTicks = SDL_GetTicks(); |  | ||||||
|  |  | ||||||
|     Object cube = Object::LoadFile("./assets/cube.obj"); |  | ||||||
|     Object monkey = Object::LoadFile("./assets/monkey.obj"); |  | ||||||
|  |  | ||||||
|     bool paused = false; |  | ||||||
|  |  | ||||||
|     bool quit = false; |  | ||||||
|     while (!quit) { |  | ||||||
|         Uint64 currentTicks = SDL_GetTicks(); |  | ||||||
|         float deltaTime = (currentTicks - lastTicks) / 1000.0f; // seconds |  | ||||||
|  |  | ||||||
|         lastTicks = currentTicks; |  | ||||||
|  |  | ||||||
|         SDL_Event event; |  | ||||||
|         while(SDL_PollEvent(&event)) { |  | ||||||
|             switch (event.type) { |  | ||||||
|                 case SDL_EVENT_WINDOW_CLOSE_REQUESTED: |  | ||||||
|                 case SDL_EVENT_QUIT: |  | ||||||
|                     quit = true; |  | ||||||
|                     break; |  | ||||||
|                 case SDL_EVENT_WINDOW_RESIZED: |  | ||||||
|                     int width, height; |  | ||||||
|                     if (SDL_GetWindowSize(window, &width, &height)) { |  | ||||||
|                         glViewport( |  | ||||||
|                             0, |  | ||||||
|                             0, |  | ||||||
|                             width, |  | ||||||
|                             height); |  | ||||||
|                     } |  | ||||||
|                     break; |  | ||||||
|                 case SDL_EVENT_KEY_DOWN: |  | ||||||
|                     switch (event.key.key) { |  | ||||||
|                     case SDLK_SPACE: |  | ||||||
|                         paused = !paused; |  | ||||||
|                         break; |  | ||||||
|                     // case SDLK_F5: |  | ||||||
|                     //     reload_shaders(&context); |  | ||||||
|                     //     break; |  | ||||||
|                     default: break; |  | ||||||
|                     }; |  | ||||||
|                     break; |  | ||||||
|                 default: break; |  | ||||||
|             }; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // update rotation |  | ||||||
|         if (!paused) { |  | ||||||
|             angle += glm::radians(45.0f) * deltaTime; // 72° per second |  | ||||||
|             if (angle > glm::two_pi<float>()) { |  | ||||||
|                 angle -= glm::two_pi<float>(); // keep value small |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // std::cout << "angle = " << angle << std::endl; |  | ||||||
|  |  | ||||||
|         glClearColor(0x18/255.0f, 0x18/255.0f, 0x18/255.0f, 1); |  | ||||||
|         glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); |  | ||||||
|  |  | ||||||
|         // Triangle render |  | ||||||
|         { |  | ||||||
|             simpleShader.use(); |  | ||||||
|  |  | ||||||
|             model = glm::rotate( |  | ||||||
|                 glm::mat4(1.f), |  | ||||||
|                 angle, |  | ||||||
|                 glm::vec3(0.8f, -0.4f, 0.5f) |  | ||||||
|             ); |  | ||||||
|  |  | ||||||
|             // lightPosition -= glm::vec3(0.05f, 0.f, 0.f) * deltaTime; |  | ||||||
|  |  | ||||||
|             simpleShader.setMat4("u_model", model); |  | ||||||
|             simpleShader.setMat4("u_view", view); |  | ||||||
|             simpleShader.setMat4("u_projection", projection); |  | ||||||
|  |  | ||||||
|             simpleShader.setVec3("lightColor", glm::vec3(1.0f, 1.0f, 1.0f)); |  | ||||||
|             simpleShader.setVec3("lightPos", lightPosition); |  | ||||||
|             simpleShader.setVec3("viewPos", cameraPosition); |  | ||||||
|             simpleShader.setFloat("ambientStrength", 0.2f); |  | ||||||
|             simpleShader.setFloat("specularStrength", 0.5f); |  | ||||||
|  |  | ||||||
|             vertices.Draw(); |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         SDL_GL_SwapWindow(window); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     SDL_GL_DestroyContext(glcontext); |  | ||||||
|     SDL_DestroyWindow(window); |  | ||||||
|  |  | ||||||
|     return 0; |  | ||||||
| } |  | ||||||
							
								
								
									
										34
									
								
								src/main.cpp
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								src/main.cpp
									
									
									
									
									
								
							| @ -1,6 +1,14 @@ | |||||||
| #include <iostream> | #include <iostream> | ||||||
|  |  | ||||||
|  | // #ifdef WIN32 | ||||||
| #define _USE_MATH_DEFINES | #define _USE_MATH_DEFINES | ||||||
| #include <cmath> | #include <cmath> | ||||||
|  | // #endif | ||||||
|  |  | ||||||
|  | #ifndef WIN32 | ||||||
|  | #define GLEW_STATIC | ||||||
|  | #endif | ||||||
|  |  | ||||||
| #include <vector> | #include <vector> | ||||||
| #include <glm/glm.hpp> | #include <glm/glm.hpp> | ||||||
| #include <glm/ext/quaternion_geometric.hpp> | #include <glm/ext/quaternion_geometric.hpp> | ||||||
| @ -9,12 +17,10 @@ | |||||||
| #include <GL/glew.h> | #include <GL/glew.h> | ||||||
| #include <SDL3/SDL.h> | #include <SDL3/SDL.h> | ||||||
|  |  | ||||||
| #include "shader.h" | #include "renderer/shader.h" | ||||||
| #include "file_manager.h" | #include "IO/file_manager.h" | ||||||
| #include "prelude.h" | #include "renderer/debug.h" | ||||||
| #include "block.h" | #include "renderer/wavefront.h" | ||||||
| #include "vertex.h" |  | ||||||
| #include "model.h" |  | ||||||
|  |  | ||||||
| #define WIDTH 1024 | #define WIDTH 1024 | ||||||
| #define HEIGHT 768 | #define HEIGHT 768 | ||||||
| @ -49,7 +55,7 @@ int main() { | |||||||
|  |  | ||||||
|     glEnable(GL_DEBUG_OUTPUT); |     glEnable(GL_DEBUG_OUTPUT); | ||||||
|     glEnable(GL_DEPTH_TEST); |     glEnable(GL_DEPTH_TEST); | ||||||
|     glDebugMessageCallback(MessageCallback, 0); |     glDebugMessageCallback(MessageCallback, nullptr); | ||||||
|  |  | ||||||
|     Shader simpleShader; |     Shader simpleShader; | ||||||
|     simpleShader.init( |     simpleShader.init( | ||||||
| @ -67,8 +73,8 @@ int main() { | |||||||
|     glm::mat4 model(1.f); |     glm::mat4 model(1.f); | ||||||
|  |  | ||||||
|     glm::mat4 projection = glm::perspective( |     glm::mat4 projection = glm::perspective( | ||||||
|         (float)M_PI_2, |         static_cast<float>(M_PI_2), | ||||||
|         (float)screenWidth / (float)screenHeight, |         static_cast<float>(screenWidth) / static_cast<float>(screenHeight), | ||||||
|         0.01f, |         0.01f, | ||||||
|         100.0f |         100.0f | ||||||
|     ); |     ); | ||||||
| @ -79,7 +85,7 @@ int main() { | |||||||
|     // Object teapot = Object::LoadFile("./assets/kastrula/kastrula.obj"); |     // Object teapot = Object::LoadFile("./assets/kastrula/kastrula.obj"); | ||||||
|     // Object bricks = Object::LoadFile("./assets/bricks/bricks.obj"); |     // Object bricks = Object::LoadFile("./assets/bricks/bricks.obj"); | ||||||
|     Object lightSource = Object::LoadFile("./assets/cube.obj"); |     Object lightSource = Object::LoadFile("./assets/cube.obj"); | ||||||
|     Object target = Object::LoadFile("./assets/car/car.obj"); |     Object target = Object::LoadFile("./assets/monkey.obj"); | ||||||
|  |  | ||||||
|     bool paused = false; |     bool paused = false; | ||||||
|  |  | ||||||
| @ -93,7 +99,7 @@ int main() { | |||||||
|     bool quit = false; |     bool quit = false; | ||||||
|     while (!quit) { |     while (!quit) { | ||||||
|         Uint64 currentTicks = SDL_GetTicks(); |         Uint64 currentTicks = SDL_GetTicks(); | ||||||
|         float deltaTime = (currentTicks - lastTicks) / 1000.0f; // seconds |         float deltaTime = static_cast<float>(currentTicks - lastTicks) / 1000.0f; // seconds | ||||||
|  |  | ||||||
|         lastTicks = currentTicks; |         lastTicks = currentTicks; | ||||||
|  |  | ||||||
| @ -127,7 +133,7 @@ int main() { | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         float mouseXRel, mouseYRel; |         float mouseXRel, mouseYRel; | ||||||
|         Uint32 mouseState = SDL_GetRelativeMouseState(&mouseXRel, &mouseYRel); |         SDL_GetRelativeMouseState(&mouseXRel, &mouseYRel); | ||||||
|  |  | ||||||
|         float sensitivity = 0.1f; // tweak as needed |         float sensitivity = 0.1f; // tweak as needed | ||||||
|         yaw   += mouseXRel * sensitivity; |         yaw   += mouseXRel * sensitivity; | ||||||
| @ -176,8 +182,6 @@ int main() { | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // std::cout << "angle = " << angle << std::endl; |  | ||||||
|  |  | ||||||
|         glClearColor(0x18/255.0f, 0x18/255.0f, 0x18/255.0f, 1); |         glClearColor(0x18/255.0f, 0x18/255.0f, 0x18/255.0f, 1); | ||||||
|         glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); |         glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); | ||||||
|  |  | ||||||
| @ -219,7 +223,7 @@ int main() { | |||||||
|         Uint64 elapsed = currentTicks - startTicks; |         Uint64 elapsed = currentTicks - startTicks; | ||||||
|  |  | ||||||
|         if (elapsed >= 1000) { // one second passed |         if (elapsed >= 1000) { // one second passed | ||||||
|             double fps = (double)frameCount / (elapsed / 1000.0); |             double fps = static_cast<double>(frameCount) / (static_cast<double>(elapsed) / 1000.0); | ||||||
|             std::cout << "FPS: " << fps << std::endl; |             std::cout << "FPS: " << fps << std::endl; | ||||||
|             frameCount = 0; |             frameCount = 0; | ||||||
|             startTicks = currentTicks; |             startTicks = currentTicks; | ||||||
|  | |||||||
							
								
								
									
										365
									
								
								src/model.cpp
									
									
									
									
									
								
							
							
						
						
									
										365
									
								
								src/model.cpp
									
									
									
									
									
								
							| @ -1,365 +0,0 @@ | |||||||
| #include <iostream> |  | ||||||
| #include <fstream> |  | ||||||
| #include <sstream> |  | ||||||
| #include <memory> |  | ||||||
| #include <filesystem> |  | ||||||
| #include <GL/glew.h> |  | ||||||
|  |  | ||||||
| #include "model.h" |  | ||||||
|  |  | ||||||
| ObjElement toElement(const std::string &s) { |  | ||||||
|     if (s == "#") return ObjElement::OHASH; |  | ||||||
|     if (s == "mtllib") return ObjElement::MTLLIB; |  | ||||||
|     if (s == "usemtl") return ObjElement::USEMTL; |  | ||||||
|     if (s == "o") return ObjElement::O; |  | ||||||
|     if (s == "v") return ObjElement::V; |  | ||||||
|     if (s == "vn") return ObjElement::VN; |  | ||||||
|     if (s == "vt") return ObjElement::VT; |  | ||||||
|     if (s == "f") return ObjElement::F; |  | ||||||
|     return ObjElement::OUNKNOWN; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| MtlElement toMtlElement(const std::string &s) { |  | ||||||
|     if (s == "#") return MtlElement::MHASH; |  | ||||||
|     if (s == "newmtl") return MtlElement::NEWMTL; |  | ||||||
|     if (s == "Ns") return MtlElement::NS; |  | ||||||
|     if (s == "Ka") return MtlElement::KA; |  | ||||||
|     if (s == "Ks") return MtlElement::KS; |  | ||||||
|     if (s == "Kd") return MtlElement::KD; |  | ||||||
|     if (s == "Ni") return MtlElement::NI; |  | ||||||
|     if (s == "d") return MtlElement::D; |  | ||||||
|     if (s == "illum") return MtlElement::ILLUM; |  | ||||||
|     if (s == "map_Kd") return MtlElement::MAP_KD; |  | ||||||
|     if (s == "map_Ka") return MtlElement::MAP_KA; |  | ||||||
|     // if (s == "map_Ke") return MtlElement::MAP_KE; |  | ||||||
|     return MtlElement::MUNKNOWN; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Vertex::DefineAttrib() |  | ||||||
| { |  | ||||||
|     glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, sizeof(Vertex), reinterpret_cast<const void*>(offsetof(Vertex, m_position))); |  | ||||||
|     glEnableVertexAttribArray(0); |  | ||||||
|  |  | ||||||
|     glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, sizeof(Vertex), reinterpret_cast<const void*>(offsetof(Vertex, m_normal))); |  | ||||||
|     glEnableVertexAttribArray(1); |  | ||||||
|      |  | ||||||
|     glVertexAttribPointer(2, 2, GL_FLOAT, GL_FALSE, sizeof(Vertex), reinterpret_cast<const void*>(offsetof(Vertex, m_texCoord))); |  | ||||||
|     glEnableVertexAttribArray(2); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Face::PushItem(const FaceItem& item) |  | ||||||
| { |  | ||||||
|     m_items.push_back(item); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| inline int Object::NormalizeIndex(const std::string &s, int baseCount) { |  | ||||||
|     if (s.empty()) return -1; |  | ||||||
|     int idx = std::stoi(s); |  | ||||||
|     if (idx > 0) return idx - 1; |  | ||||||
|     return baseCount + idx; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| Mesh::Mesh() { |  | ||||||
|     glGenVertexArrays(1, &m_vao); |  | ||||||
|     glGenBuffers(1, &m_vbo); |  | ||||||
|     glGenBuffers(1, &m_ebo); |  | ||||||
|  |  | ||||||
|     glBindVertexArray(m_vao); |  | ||||||
|  |  | ||||||
|     // VBO (vertex buffer) |  | ||||||
|     glBindBuffer(GL_ARRAY_BUFFER, m_vbo); |  | ||||||
|     glBufferData(GL_ARRAY_BUFFER, 0, nullptr, GL_STATIC_DRAW); |  | ||||||
|  |  | ||||||
|     // EBO (index buffer) |  | ||||||
|     glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_ebo); |  | ||||||
|     glBufferData(GL_ELEMENT_ARRAY_BUFFER, 0, nullptr, GL_STATIC_DRAW); |  | ||||||
|  |  | ||||||
|     Vertex::DefineAttrib(); |  | ||||||
|  |  | ||||||
|     glBindBuffer(GL_ARRAY_BUFFER, 0); |  | ||||||
|     glBindVertexArray(0); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| Object::Object() { |  | ||||||
|     m_vertices = std::vector<glm::vec3>(); |  | ||||||
|     m_normals = std::vector<glm::vec3>(); |  | ||||||
|     m_texCoords = std::vector<glm::vec2>(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Object::LoadMaterials(const std::string& filename) { |  | ||||||
|     std::ifstream file(filename); |  | ||||||
|  |  | ||||||
|     std::string currentMaterialName; |  | ||||||
|     std::shared_ptr<Material> currentMaterial; |  | ||||||
|      |  | ||||||
|     std::string line; |  | ||||||
|     while (std::getline(file, line)) { |  | ||||||
|         std::istringstream iss(line); |  | ||||||
|         std::string prefix; |  | ||||||
|         iss >> prefix; |  | ||||||
|         switch(toMtlElement(prefix)) { |  | ||||||
|         case MtlElement::MHASH: |  | ||||||
|         { |  | ||||||
|             std::cout << "comment: " << line << std::endl; |  | ||||||
|             continue; |  | ||||||
|         } |  | ||||||
|         case MtlElement::NEWMTL: |  | ||||||
|         { |  | ||||||
|             if (currentMaterial) { |  | ||||||
|                 m_materials.insert(std::make_pair(currentMaterialName, std::move(currentMaterial))); |  | ||||||
|                 currentMaterial = nullptr; |  | ||||||
|             } |  | ||||||
|             std::string materialName; |  | ||||||
|             iss >> materialName; |  | ||||||
|             currentMaterialName = materialName; |  | ||||||
|             currentMaterial = std::make_shared<Material>(); |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case MtlElement::NS: |  | ||||||
|         { |  | ||||||
|             float weight; |  | ||||||
|             iss >> weight; |  | ||||||
|             currentMaterial->SetSpecularWeight(weight); |  | ||||||
|         } |  | ||||||
|         case MtlElement::KA: |  | ||||||
|         { |  | ||||||
|             float r, g, b; |  | ||||||
|             iss >> r >> g >> b; |  | ||||||
|             currentMaterial->SetAmbientColor(glm::vec3(r, g, b)); |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case MtlElement::KS: |  | ||||||
|         { |  | ||||||
|             float r, g, b; |  | ||||||
|             iss >> r >> g >> b; |  | ||||||
|             currentMaterial->SetSpecularColor(glm::vec3(r, g, b)); |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case MtlElement::KD: |  | ||||||
|         { |  | ||||||
|             float r, g, b; |  | ||||||
|             iss >> r >> g >> b; |  | ||||||
|             currentMaterial->SetDiffuseColor(glm::vec3(r, g, b)); |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case MtlElement::MAP_KD: |  | ||||||
|         { |  | ||||||
|             std::string texturePath; |  | ||||||
|             std::string part; |  | ||||||
|             while (iss >> part) { |  | ||||||
|                 texturePath += part + " "; |  | ||||||
|             } |  | ||||||
|             texturePath = texturePath.substr(0, texturePath.size() - 1); |  | ||||||
|             currentMaterial->SetDiffuseTexture(Texture::LoadFile(texturePath)); |  | ||||||
|         } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (currentMaterial) { |  | ||||||
|         // m_materials.insert(std::make_pair(currentMaterialName, std::move(currentMaterial))); |  | ||||||
|         AddMaterial(currentMaterialName, std::move(currentMaterial)); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Object::AddMaterial(std::string name, std::shared_ptr<Material> material) |  | ||||||
| { |  | ||||||
|     m_materials.insert(std::make_pair(std::move(name), std::move(material))); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| std::shared_ptr<Material> Object::GetMaterial(std::string name) |  | ||||||
| { |  | ||||||
|     auto material = m_materials.find(name); |  | ||||||
|     if (material == m_materials.end()) return nullptr; |  | ||||||
|     return material->second; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Object::CreateNewMesh(const std::string& materialName) |  | ||||||
| { |  | ||||||
|     Mesh mesh; |  | ||||||
|     mesh.materialName = materialName; |  | ||||||
|     m_meshes.push_back(mesh); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| Mesh& Object::GetLastMesh() |  | ||||||
| { |  | ||||||
|     if (m_meshes.empty()) { |  | ||||||
|         auto material = std::make_shared<Material>(); |  | ||||||
|         material->SetAmbientColor(glm::vec3(0.52f, 0.52f, 0.52f)); |  | ||||||
|         // TODO: come up with name for a default material |  | ||||||
|         AddMaterial("", std::move(material)); |  | ||||||
|         CreateNewMesh(""); |  | ||||||
|     } |  | ||||||
|     return m_meshes.back(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| Object Object::LoadFile(const std::string& filename) { |  | ||||||
|     std::ifstream file(filename); |  | ||||||
|  |  | ||||||
|     Object obj; |  | ||||||
|  |  | ||||||
|     std::string line; |  | ||||||
|     while (std::getline(file, line)) { |  | ||||||
|         std::istringstream iss(line); |  | ||||||
|         std::string prefix; |  | ||||||
|         iss >> prefix; |  | ||||||
|         switch(toElement(prefix)) { |  | ||||||
|         // comment |  | ||||||
|         case ObjElement::OHASH: |  | ||||||
|         { |  | ||||||
|             std::cout << "comment: " << line << std::endl; |  | ||||||
|             continue; |  | ||||||
|         } |  | ||||||
|         case ObjElement::MTLLIB: |  | ||||||
|         { |  | ||||||
|             std::string mtlFile; |  | ||||||
|             iss >> mtlFile; |  | ||||||
|             std::filesystem::path fullPath = filename; |  | ||||||
|             std::filesystem::path mtlPath = fullPath.replace_filename(mtlFile); |  | ||||||
|             obj.LoadMaterials(mtlPath.u8string()); |  | ||||||
|             std::cout << "loaded mtl at '" << mtlPath << "' with " |  | ||||||
|                 << obj.m_materials.size() << " materials" << std::endl; |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case ObjElement::USEMTL: |  | ||||||
|         { |  | ||||||
|             std::string materialName; |  | ||||||
|             iss >> materialName; |  | ||||||
|             auto& mesh = obj.GetLastMesh(); |  | ||||||
|             if (mesh.materialName != materialName) { |  | ||||||
|                 Mesh mesh; |  | ||||||
|                 mesh.materialName = materialName; |  | ||||||
|                 obj.m_meshes.push_back(mesh); |  | ||||||
|             } |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         // object name I suppose |  | ||||||
|         case ObjElement::O: |  | ||||||
|         { |  | ||||||
|             obj.m_name = line.substr(2); |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         // vertex with its position |  | ||||||
|         case ObjElement::V: |  | ||||||
|         { |  | ||||||
|             float x, y, z, w; |  | ||||||
|             w = 1.0f; |  | ||||||
|             iss >> x >> y >> z; |  | ||||||
|             if (iss >> w) { |  | ||||||
|                 x /= w; |  | ||||||
|                 y /= w; |  | ||||||
|                 z /= w; |  | ||||||
|             } |  | ||||||
|             obj.m_vertices.emplace_back(x, y, z); |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case ObjElement::VN: |  | ||||||
|         { |  | ||||||
|             float x, y, z; |  | ||||||
|             iss >> x >> y >> z; |  | ||||||
|             obj.m_normals.emplace_back(x, y, z); |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case ObjElement::VT: |  | ||||||
|         { |  | ||||||
|             float u, v; |  | ||||||
|             iss >> u >> v; |  | ||||||
|             obj.m_texCoords.emplace_back(u, 1.0f - v); |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         case ObjElement::F: |  | ||||||
|         { |  | ||||||
|             auto& mesh = obj.GetLastMesh(); |  | ||||||
|             std::string token; |  | ||||||
|              |  | ||||||
|             Face fv; |  | ||||||
|  |  | ||||||
|             while (iss >> token) { |  | ||||||
|                 std::string a, b, c; |  | ||||||
|                 std::istringstream ref(token); |  | ||||||
|  |  | ||||||
|                 std::getline(ref, a, '/'); |  | ||||||
|                 std::getline(ref, b, '/'); |  | ||||||
|                 std::getline(ref, c, '/'); |  | ||||||
|  |  | ||||||
|                 int vi = Object::NormalizeIndex(a, (int)obj.m_vertices.size()); |  | ||||||
|                 int ti = Object::NormalizeIndex(b, (int)obj.m_texCoords.size()); |  | ||||||
|                 int ni = Object::NormalizeIndex(c, (int)obj.m_normals.size()); |  | ||||||
|  |  | ||||||
|                 glm::vec3 vert, norm; |  | ||||||
|                 glm::vec2 texCoord; |  | ||||||
|                 vert = obj.m_vertices[vi]; |  | ||||||
|                 if (ni >= 0) norm = obj.m_normals[ni]; |  | ||||||
|                 if (ti >= 0) texCoord = obj.m_texCoords[ti]; |  | ||||||
|  |  | ||||||
|                 mesh.m_vertexBuffer.emplace_back(vert, norm, texCoord); |  | ||||||
|                 mesh.m_indexBuffer.push_back(mesh.m_vertexBuffer.size() - 1); |  | ||||||
|             } |  | ||||||
|              |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     std::cout << "Object name: " << obj.m_name << std::endl; |  | ||||||
|     std::cout << "Vertices count: " << obj.m_vertices.size() << std::endl; |  | ||||||
|     std::cout << "Normals count: " << obj.m_normals.size() << std::endl; |  | ||||||
|     std::cout << "TexCoords count: " << obj.m_texCoords.size() << std::endl; |  | ||||||
|     std::cout << "Meshes count: " << obj.m_meshes.size() << std::endl; |  | ||||||
|     std::cout << "Materials count: " << obj.m_materials.size() << std::endl; |  | ||||||
|      |  | ||||||
|     // std::cout << "Vertex Buffer size: " << obj.m_vertexBuffer.size() << std::endl; |  | ||||||
|     // std::cout << "Index Buffer size: " << obj.m_indexBuffer.size() << std::endl; |  | ||||||
|  |  | ||||||
|     file.close(); |  | ||||||
|  |  | ||||||
|     for (auto &mesh : obj.m_meshes) { |  | ||||||
|         mesh.Upload(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return obj; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Object::Render(Shader& shader) |  | ||||||
| { |  | ||||||
|     for (auto &mesh : m_meshes) { |  | ||||||
|         auto material = GetMaterial(mesh.materialName); |  | ||||||
|          |  | ||||||
|         shader.setFloat("ambientStrength", 0.2f); |  | ||||||
|         shader.setFloat("specularStrength", material->GetSpecularWeight()); |  | ||||||
|         shader.setVec3("ambientColor", material->GetAmbientColor()); |  | ||||||
|         shader.setVec3("diffuseColor", material->GetDiffuseColor()); |  | ||||||
|         shader.setVec3("specularColor", material->GetSpecularColor()); |  | ||||||
|  |  | ||||||
|         if (material->HasDiffuseTexture()) { |  | ||||||
|             shader.setBool("useTexture", true); |  | ||||||
|             glActiveTexture(GL_TEXTURE0); |  | ||||||
|             glBindTexture(GL_TEXTURE_2D, material->GetDiffuseTexture()->GetID()); |  | ||||||
|             shader.setInt("diffuseTex", 0); |  | ||||||
|         } else { |  | ||||||
|             shader.setBool("useTexture", false); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         mesh.Render(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Mesh::Upload() |  | ||||||
| { |  | ||||||
|     glBindVertexArray(m_vao); |  | ||||||
|  |  | ||||||
|     glBindBuffer(GL_ARRAY_BUFFER, m_vbo); |  | ||||||
|     glBufferData(GL_ARRAY_BUFFER, m_vertexBuffer.size() * sizeof(Vertex), m_vertexBuffer.data(), GL_STATIC_DRAW); |  | ||||||
|  |  | ||||||
|     // Upload indices |  | ||||||
|     glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_ebo); |  | ||||||
|     glBufferData(GL_ELEMENT_ARRAY_BUFFER, m_indexBuffer.size() * sizeof(unsigned int), m_indexBuffer.data(), GL_STATIC_DRAW); |  | ||||||
|  |  | ||||||
|     glBindVertexArray(0); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Mesh::Render() |  | ||||||
| { |  | ||||||
|     Bind(); |  | ||||||
|     glDrawElements(GL_TRIANGLES, static_cast<GLsizei>(m_indexBuffer.size()), GL_UNSIGNED_INT, 0); |  | ||||||
|     Unbind(); |  | ||||||
| } |  | ||||||
							
								
								
									
										190
									
								
								src/prelude.cpp
									
									
									
									
									
								
							
							
						
						
									
										190
									
								
								src/prelude.cpp
									
									
									
									
									
								
							| @ -1,190 +0,0 @@ | |||||||
| #include <stdio.h> |  | ||||||
| #include <stdlib.h> |  | ||||||
| #include <stdarg.h> |  | ||||||
| #include <stdbool.h> |  | ||||||
| #include <string.h> |  | ||||||
| #include <errno.h> |  | ||||||
| #include "prelude.h" |  | ||||||
|  |  | ||||||
| // #define GLEW_STATIC |  | ||||||
| #include <GL/glew.h> |  | ||||||
|  |  | ||||||
| #define SCREEN_WIDTH 1024 |  | ||||||
| #define SCREEN_HEIGHT 768 |  | ||||||
|  |  | ||||||
| void panic_errno(const char *fmt, ...) |  | ||||||
| { |  | ||||||
|     fprintf(stderr, "ERROR: "); |  | ||||||
|  |  | ||||||
|     va_list args; |  | ||||||
|     va_start(args, fmt); |  | ||||||
|     vfprintf(stderr, fmt, args); |  | ||||||
|     va_end(args); |  | ||||||
|  |  | ||||||
|     fprintf(stderr, ": %s\n", strerror(errno)); |  | ||||||
|  |  | ||||||
|     exit(1); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| char *slurp_file(const char *file_path) |  | ||||||
| { |  | ||||||
| #define SLURP_FILE_PANIC panic_errno("Could not read file `%s`", file_path) |  | ||||||
|     FILE *f = fopen(file_path, "r"); |  | ||||||
|     if (f == NULL) SLURP_FILE_PANIC; |  | ||||||
|     if (fseek(f, 0, SEEK_END) < 0) SLURP_FILE_PANIC; |  | ||||||
|  |  | ||||||
|     long size = ftell(f); |  | ||||||
|     if (size < 0) SLURP_FILE_PANIC; |  | ||||||
|  |  | ||||||
|     char *buffer = (char*)malloc(size + 1); |  | ||||||
|     if (buffer == NULL) SLURP_FILE_PANIC; |  | ||||||
|  |  | ||||||
|     if (fseek(f, 0, SEEK_SET) < 0) SLURP_FILE_PANIC; |  | ||||||
|  |  | ||||||
|     fread(buffer, 1, size, f); |  | ||||||
|     if (ferror(f) < 0) SLURP_FILE_PANIC; |  | ||||||
|  |  | ||||||
|     buffer[size] = '\0'; |  | ||||||
|  |  | ||||||
|     if (fclose(f) < 0) SLURP_FILE_PANIC; |  | ||||||
|  |  | ||||||
|     return buffer; |  | ||||||
| #undef SLURP_FILE_PANIC |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool compile_shader_source(const GLchar *source, GLenum shader_type, GLuint *shader) |  | ||||||
| { |  | ||||||
|     *shader = glCreateShader(shader_type); |  | ||||||
|     glShaderSource(*shader, 1, &source, NULL); |  | ||||||
|     glCompileShader(*shader); |  | ||||||
|  |  | ||||||
|     GLint compiled = 0; |  | ||||||
|     glGetShaderiv(*shader, GL_COMPILE_STATUS, &compiled); |  | ||||||
|  |  | ||||||
|     if (!compiled) { |  | ||||||
|         GLchar message[1024]; |  | ||||||
|         GLsizei message_size = 0; |  | ||||||
|         glGetShaderInfoLog(*shader, sizeof(message), &message_size, message); |  | ||||||
|         fprintf(stderr, "%.*s\n", message_size, message); |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return true; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool compile_shader_file(const char *file_path, GLenum shader_type, GLuint *shader) |  | ||||||
| { |  | ||||||
|     char *source = slurp_file(file_path); |  | ||||||
|     bool err = compile_shader_source(source, shader_type, shader); |  | ||||||
|     free(source); |  | ||||||
|     return err; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool link_program(GLuint vert_shader, GLuint frag_shader, GLuint *program) |  | ||||||
| { |  | ||||||
|     *program = glCreateProgram(); |  | ||||||
|  |  | ||||||
|     glAttachShader(*program, vert_shader); |  | ||||||
|     glAttachShader(*program, frag_shader); |  | ||||||
|     glLinkProgram(*program); |  | ||||||
|  |  | ||||||
|     GLint linked = 0; |  | ||||||
|     glGetProgramiv(*program, GL_LINK_STATUS, &linked); |  | ||||||
|     if (!linked) { |  | ||||||
|         GLsizei message_size = 0; |  | ||||||
|         GLchar message[1024]; |  | ||||||
|  |  | ||||||
|         glGetProgramInfoLog(*program, sizeof(message), &message_size, message); |  | ||||||
|         fprintf(stderr, "Program Linking: %.*s\n", message_size, message); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     glDeleteShader(vert_shader); |  | ||||||
|     glDeleteShader(frag_shader); |  | ||||||
|  |  | ||||||
|     return program; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void reload_shaders(RenderContext* context) |  | ||||||
| { |  | ||||||
|     glDeleteProgram(context->program); |  | ||||||
|  |  | ||||||
|     context->program_failed = false; |  | ||||||
|     glClearColor(0.0f, 0.0f, 0.0f, 0.0f); |  | ||||||
|  |  | ||||||
|     GLuint vert = 0; |  | ||||||
|     if (!compile_shader_file("./main.vert", GL_VERTEX_SHADER, &vert)) { |  | ||||||
|         glClearColor(1.0f, 0.0f, 0.0f, 1.0f); |  | ||||||
|         context->program_failed = true; |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     GLuint frag = 0; |  | ||||||
|     if (!compile_shader_file("./main.frag", GL_FRAGMENT_SHADER, &frag)) { |  | ||||||
|         glClearColor(1.0f, 0.0f, 0.0f, 1.0f); |  | ||||||
|         context->program_failed = true; |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (!link_program(vert, frag, &context->program)) { |  | ||||||
|         glClearColor(1.0f, 0.0f, 0.0f, 1.0f); |  | ||||||
|         context->program_failed = true; |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     glUseProgram(context->program); |  | ||||||
|  |  | ||||||
|     context->resolution_location = glGetUniformLocation(context->program, "resolution"); |  | ||||||
|     context->time_location = glGetUniformLocation(context->program, "time"); |  | ||||||
|  |  | ||||||
|     printf("Successfully Reload the Shaders\n"); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void window_size_callback(SDL_Window* window, int width, int height) |  | ||||||
| { |  | ||||||
|     (void) window; |  | ||||||
|     glViewport( |  | ||||||
|         width / 2 - SCREEN_WIDTH / 2, |  | ||||||
|         height / 2 - SCREEN_HEIGHT / 2, |  | ||||||
|         SCREEN_WIDTH, |  | ||||||
|         SCREEN_HEIGHT); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void MessageCallback(GLenum source, |  | ||||||
|                      GLenum type, |  | ||||||
|                      GLuint id, |  | ||||||
|                      GLenum severity, |  | ||||||
|                      GLsizei length, |  | ||||||
|                      const GLchar* message, |  | ||||||
|                      const void* userParam) |  | ||||||
| { |  | ||||||
|     (void) source; |  | ||||||
|     (void) id; |  | ||||||
|     (void) length; |  | ||||||
|     (void) userParam; |  | ||||||
|     fprintf(stderr, "GL CALLBACK: %s type = 0x%x, severity = 0x%x, message = %s\n", |  | ||||||
|             (type == GL_DEBUG_TYPE_ERROR ? "** GL ERROR **" : ""), |  | ||||||
|             type, severity, message); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void process_prelude(RenderContext *context) |  | ||||||
| { |  | ||||||
|     // glfwSetKeyCallback(window, key_callback); |  | ||||||
|     // glfwSetFramebufferSizeCallback(window, window_size_callback); |  | ||||||
|  |  | ||||||
|     glClear(GL_COLOR_BUFFER_BIT); |  | ||||||
|  |  | ||||||
|     if (!context->program_failed) { |  | ||||||
|         glUniform2f(context->resolution_location, |  | ||||||
|                     SCREEN_WIDTH, |  | ||||||
|                     SCREEN_HEIGHT); |  | ||||||
|         glUniform1f(context->time_location, context->time); |  | ||||||
|  |  | ||||||
|         glDrawArrays(GL_TRIANGLE_STRIP, 0, 4); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     int64_t cur_time = SDL_GetTicks(); |  | ||||||
|     if (!context->pause) { |  | ||||||
|         context->time += cur_time - context->prev_time; |  | ||||||
|     } |  | ||||||
|     context->prev_time = cur_time; |  | ||||||
| } |  | ||||||
							
								
								
									
										15
									
								
								src/renderer/basics.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								src/renderer/basics.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,15 @@ | |||||||
|  | #include <GL/glew.h> | ||||||
|  |  | ||||||
|  | #include "renderer/basics.h" | ||||||
|  |  | ||||||
|  | void Vertex::DefineAttrib() | ||||||
|  | { | ||||||
|  |     glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, sizeof(Vertex), reinterpret_cast<const void*>(offsetof(Vertex, m_position))); | ||||||
|  |     glEnableVertexAttribArray(0); | ||||||
|  |  | ||||||
|  |     glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, sizeof(Vertex), reinterpret_cast<const void*>(offsetof(Vertex, m_normal))); | ||||||
|  |     glEnableVertexAttribArray(1); | ||||||
|  |      | ||||||
|  |     glVertexAttribPointer(2, 2, GL_FLOAT, GL_FALSE, sizeof(Vertex), reinterpret_cast<const void*>(offsetof(Vertex, m_texCoord))); | ||||||
|  |     glEnableVertexAttribArray(2); | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								src/renderer/debug.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/renderer/debug.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | |||||||
|  | #include "renderer/debug.h" | ||||||
|  |  | ||||||
|  | void MessageCallback(GLenum source, | ||||||
|  |                      GLenum type, | ||||||
|  |                      GLuint id, | ||||||
|  |                      GLenum severity, | ||||||
|  |                      GLsizei length, | ||||||
|  |                      const GLchar* message, | ||||||
|  |                      const void* userParam) | ||||||
|  | { | ||||||
|  |     (void) source; | ||||||
|  |     (void) id; | ||||||
|  |     (void) length; | ||||||
|  |     (void) userParam; | ||||||
|  |     std::cerr << "GL CALLBACK: " << (type == GL_DEBUG_TYPE_ERROR ? "** GL ERROR **" : "") | ||||||
|  |               << " type = 0x" << type | ||||||
|  |               << ", severity = 0x" << severity | ||||||
|  |               << ", message = " << message << std::endl; | ||||||
|  |     // std::cerr << "GL CALLBACK: %s type = 0x%x, severity = 0x%x, message = %s\n", | ||||||
|  |     //         (type == GL_DEBUG_TYPE_ERROR ? "** GL ERROR **" : ""), | ||||||
|  |     //         type, severity, message); | ||||||
|  | } | ||||||
							
								
								
									
										43
									
								
								src/renderer/mesh.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								src/renderer/mesh.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | |||||||
|  | #include "renderer/mesh.h" | ||||||
|  |  | ||||||
|  | Mesh::Mesh() { | ||||||
|  |     glGenVertexArrays(1, &m_vao); | ||||||
|  |     glGenBuffers(1, &m_vbo); | ||||||
|  |     glGenBuffers(1, &m_ebo); | ||||||
|  |  | ||||||
|  |     glBindVertexArray(m_vao); | ||||||
|  |  | ||||||
|  |     // VBO (vertex buffer) | ||||||
|  |     glBindBuffer(GL_ARRAY_BUFFER, m_vbo); | ||||||
|  |     glBufferData(GL_ARRAY_BUFFER, 0, nullptr, GL_STATIC_DRAW); | ||||||
|  |  | ||||||
|  |     // EBO (index buffer) | ||||||
|  |     glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_ebo); | ||||||
|  |     glBufferData(GL_ELEMENT_ARRAY_BUFFER, 0, nullptr, GL_STATIC_DRAW); | ||||||
|  |  | ||||||
|  |     Vertex::DefineAttrib(); | ||||||
|  |  | ||||||
|  |     glBindBuffer(GL_ARRAY_BUFFER, 0); | ||||||
|  |     glBindVertexArray(0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Mesh::Upload() | ||||||
|  | { | ||||||
|  |     glBindVertexArray(m_vao); | ||||||
|  |  | ||||||
|  |     glBindBuffer(GL_ARRAY_BUFFER, m_vbo); | ||||||
|  |     glBufferData(GL_ARRAY_BUFFER, m_vertexBuffer.size() * sizeof(Vertex), m_vertexBuffer.data(), GL_STATIC_DRAW); | ||||||
|  |  | ||||||
|  |     // Upload indices | ||||||
|  |     glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_ebo); | ||||||
|  |     glBufferData(GL_ELEMENT_ARRAY_BUFFER, m_indexBuffer.size() * sizeof(unsigned int), m_indexBuffer.data(), GL_STATIC_DRAW); | ||||||
|  |  | ||||||
|  |     glBindVertexArray(0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Mesh::Render() | ||||||
|  | { | ||||||
|  |     Bind(); | ||||||
|  |     glDrawElements(GL_TRIANGLES, static_cast<GLsizei>(m_indexBuffer.size()), GL_UNSIGNED_INT, 0); | ||||||
|  |     Unbind(); | ||||||
|  | } | ||||||
| @ -1,4 +1,4 @@ | |||||||
| #include "shader.h" | #include "renderer/shader.h" | ||||||
| 
 | 
 | ||||||
| Shader::Shader() | Shader::Shader() | ||||||
| { | { | ||||||
| @ -1,7 +1,8 @@ | |||||||
| #include <iostream> | #include <iostream> | ||||||
| #include <memory> | #include <memory> | ||||||
|  | 
 | ||||||
| #include <GL/glew.h> | #include <GL/glew.h> | ||||||
| #include "texture.h" | #include "renderer/texture.h" | ||||||
| 
 | 
 | ||||||
| #define STB_IMAGE_IMPLEMENTATION | #define STB_IMAGE_IMPLEMENTATION | ||||||
| #include "stb_image.h" | #include "stb_image.h" | ||||||
| @ -10,7 +11,7 @@ std::unique_ptr<Texture> Texture::LoadFile(const std::string& filename) { | |||||||
|     auto texture = std::make_unique<Texture>(); |     auto texture = std::make_unique<Texture>(); | ||||||
| 
 | 
 | ||||||
|     int w, h, c; |     int w, h, c; | ||||||
|     unsigned char *data = stbi_load(filename.c_str(), &w, &h, &c, 0); |     unsigned char *data = stbi_load(filename.c_str(), &w, &h, &c, 4); | ||||||
|     if (!data) { |     if (!data) { | ||||||
|         std::cerr << "ERROR: Failed to load texture under '" << filename << "'" << std::endl; |         std::cerr << "ERROR: Failed to load texture under '" << filename << "'" << std::endl; | ||||||
|         std::exit(1); |         std::exit(1); | ||||||
| @ -25,8 +26,7 @@ std::unique_ptr<Texture> Texture::LoadFile(const std::string& filename) { | |||||||
|     glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR); |     glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR); | ||||||
|     glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); |     glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); | ||||||
| 
 | 
 | ||||||
|     // TODO: automatically detect values for this function
 |     glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA, GL_UNSIGNED_BYTE, data); | ||||||
|     glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, w, h, 0, GL_RGB, GL_UNSIGNED_BYTE, data); |  | ||||||
|     glGenerateMipmap(GL_TEXTURE_2D); |     glGenerateMipmap(GL_TEXTURE_2D); | ||||||
| 
 | 
 | ||||||
|     std::cout << "Loaded texture under '" << filename << "' with size of " << sizeof(data) << " bytes" << std::endl; |     std::cout << "Loaded texture under '" << filename << "' with size of " << sizeof(data) << " bytes" << std::endl; | ||||||
							
								
								
									
										423
									
								
								src/renderer/wavefront.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										423
									
								
								src/renderer/wavefront.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,423 @@ | |||||||
|  | #include <iostream> | ||||||
|  | #include <fstream> | ||||||
|  | #include <cstring> | ||||||
|  | #include <memory> | ||||||
|  | #include <filesystem> | ||||||
|  | #include <GL/glew.h> | ||||||
|  |  | ||||||
|  | #include "IO/parser.h" | ||||||
|  | #include "renderer/mesh.h" | ||||||
|  | #include "renderer/wavefront.h" | ||||||
|  |  | ||||||
|  | #define DEFAULT_MATERIAL_NAME "default" | ||||||
|  |  | ||||||
|  | // ObjElement toElement(const std::string &s) { | ||||||
|  | //     if (s == "#") return ObjElement::OHASH; | ||||||
|  | //     if (s == "mtllib") return ObjElement::MTLLIB; | ||||||
|  | //     if (s == "usemtl") return ObjElement::USEMTL; | ||||||
|  | //     if (s == "o") return ObjElement::O; | ||||||
|  | //     if (s == "v") return ObjElement::V; | ||||||
|  | //     if (s == "vn") return ObjElement::VN; | ||||||
|  | //     if (s == "vt") return ObjElement::VT; | ||||||
|  | //     if (s == "f") return ObjElement::F; | ||||||
|  | //     return ObjElement::OUNKNOWN; | ||||||
|  | // } | ||||||
|  |  | ||||||
|  | inline ObjElement toElement(const char* s) { | ||||||
|  |     switch (s[0]) { | ||||||
|  |         case '#': return ObjElement::OHASH; | ||||||
|  |         case 'm': if (strcmp(s, "mtllib") == 0) return ObjElement::MTLLIB; break; | ||||||
|  |         case 'u': if (strcmp(s, "usemtl") == 0) return ObjElement::USEMTL; break; | ||||||
|  |         case 'o': if (s[1] == '\0') return ObjElement::O; break; | ||||||
|  |         case 'v': | ||||||
|  |             if (s[1] == '\0') return ObjElement::V; | ||||||
|  |             if (s[1] == 'n' && s[2] == '\0') return ObjElement::VN; | ||||||
|  |             if (s[1] == 't' && s[2] == '\0') return ObjElement::VT; | ||||||
|  |             break; | ||||||
|  |         case 'f': if (s[1] == '\0') return ObjElement::F; break; | ||||||
|  |     } | ||||||
|  |     return ObjElement::OUNKNOWN; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // MtlElement toMtlElement(const std::string &s) { | ||||||
|  | //     if (s == "#") return MtlElement::MHASH; | ||||||
|  | //     if (s == "newmtl") return MtlElement::NEWMTL; | ||||||
|  | //     if (s == "Ns") return MtlElement::NS; | ||||||
|  | //     if (s == "Ka") return MtlElement::KA; | ||||||
|  | //     if (s == "Ks") return MtlElement::KS; | ||||||
|  | //     if (s == "Kd") return MtlElement::KD; | ||||||
|  | //     if (s == "Ni") return MtlElement::NI; | ||||||
|  | //     if (s == "d") return MtlElement::D; | ||||||
|  | //     if (s == "illum") return MtlElement::ILLUM; | ||||||
|  | //     if (s == "map_Kd") return MtlElement::MAP_KD; | ||||||
|  | //     if (s == "map_Ka") return MtlElement::MAP_KA; | ||||||
|  | //     // if (s == "map_Ke") return MtlElement::MAP_KE; | ||||||
|  | //     return MtlElement::MUNKNOWN; | ||||||
|  | // } | ||||||
|  |  | ||||||
|  | inline MtlElement toMtlElement(const char* s) { | ||||||
|  |     switch (s[0]) { | ||||||
|  |         case '#': return MtlElement::MHASH; | ||||||
|  |         case 'n': | ||||||
|  |             if (strcmp(s, "newmtl") == 0) return MtlElement::NEWMTL; | ||||||
|  |             break; | ||||||
|  |         case 'N': | ||||||
|  |             if (s[1] == 's' && s[2] == '\0') return MtlElement::NS; | ||||||
|  |             if (s[1] == 'i' && s[2] == '\0') return MtlElement::NI; | ||||||
|  |             break; | ||||||
|  |         case 'K': | ||||||
|  |             if (s[1] == 'a' && s[2] == '\0') return MtlElement::KA; | ||||||
|  |             if (s[1] == 's' && s[2] == '\0') return MtlElement::KS; | ||||||
|  |             if (s[1] == 'd' && s[2] == '\0') return MtlElement::KD; | ||||||
|  |             break; | ||||||
|  |         case 'd': | ||||||
|  |             if (s[1] == '\0') return MtlElement::D; | ||||||
|  |             break; | ||||||
|  |         case 'i': | ||||||
|  |             if (strcmp(s, "illum") == 0) return MtlElement::ILLUM; | ||||||
|  |             break; | ||||||
|  |         case 'm': | ||||||
|  |             if (strcmp(s, "map_Kd") == 0) return MtlElement::MAP_KD; | ||||||
|  |             if (strcmp(s, "map_Ka") == 0) return MtlElement::MAP_KA; | ||||||
|  |             // if (strcmp(s, "map_Ke") == 0) return MtlElement::MAP_KE; | ||||||
|  |             break; | ||||||
|  |     } | ||||||
|  |     return MtlElement::MUNKNOWN; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | inline int Object::NormalizeIndex(int idx, int baseCount) { | ||||||
|  |     // idx is the raw value returned by parser: | ||||||
|  |     //  0 -> means "not present" or invalid in our convention | ||||||
|  |     // >0 -> 1-based index -> convert to 0-based | ||||||
|  |     // <0 -> negative index -> relative to baseCount: baseCount + idx | ||||||
|  |     if (idx == 0) return -1;          // absent / invalid | ||||||
|  |     if (idx > 0) return idx - 1;      // 1-based -> 0-based | ||||||
|  |     return baseCount + idx;           // negative -> count from end | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Object::Object() { | ||||||
|  |     m_vertices = std::vector<glm::vec3>(); | ||||||
|  |     m_normals = std::vector<glm::vec3>(); | ||||||
|  |     m_texCoords = std::vector<glm::vec2>(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Object::LoadMaterials(const std::filesystem::path& filename) { | ||||||
|  |     std::ifstream file(filename); | ||||||
|  |     if (!file.is_open()) { | ||||||
|  |         std::cerr << "Failed to open MTL file: " << filename << std::endl; | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     std::string currentMaterialName; | ||||||
|  |     std::shared_ptr<Material> currentMaterial; | ||||||
|  |  | ||||||
|  |     char line[1024]; // buffer per line | ||||||
|  |  | ||||||
|  |     while (file.getline(line, sizeof(line))) { | ||||||
|  |         Parser p(line); | ||||||
|  |         char* prefix = p.TakeWord(); | ||||||
|  |         if (!prefix) continue; | ||||||
|  |  | ||||||
|  |         switch (toMtlElement(prefix)) { | ||||||
|  |         case MtlElement::MHASH: // comment | ||||||
|  |             continue; | ||||||
|  |  | ||||||
|  |         case MtlElement::NEWMTL: | ||||||
|  |         { | ||||||
|  |             // If a material was being built, commit it first | ||||||
|  |             if (currentMaterial) { | ||||||
|  |                 AddMaterial(currentMaterialName, std::move(currentMaterial)); | ||||||
|  |                 currentMaterial = nullptr; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             char* materialName = p.TakeWord(); | ||||||
|  |             if (materialName) { | ||||||
|  |                 currentMaterialName = materialName; | ||||||
|  |                 currentMaterial = std::make_shared<Material>(); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case MtlElement::NS: // specular weight | ||||||
|  |         { | ||||||
|  |             float weight = p.TakeFloat(); | ||||||
|  |             if (currentMaterial) currentMaterial->SetSpecularWeight(weight); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case MtlElement::KA: // ambient color | ||||||
|  |         { | ||||||
|  |             float r = p.TakeFloat(); | ||||||
|  |             float g = p.TakeFloat(); | ||||||
|  |             float b = p.TakeFloat(); | ||||||
|  |             if (currentMaterial) currentMaterial->SetAmbientColor(glm::vec3(r, g, b)); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case MtlElement::KS: // specular color | ||||||
|  |         { | ||||||
|  |             float r = p.TakeFloat(); | ||||||
|  |             float g = p.TakeFloat(); | ||||||
|  |             float b = p.TakeFloat(); | ||||||
|  |             if (currentMaterial) currentMaterial->SetSpecularColor(glm::vec3(r, g, b)); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case MtlElement::KD: // diffuse color | ||||||
|  |         { | ||||||
|  |             float r = p.TakeFloat(); | ||||||
|  |             float g = p.TakeFloat(); | ||||||
|  |             float b = p.TakeFloat(); | ||||||
|  |             if (currentMaterial) currentMaterial->SetDiffuseColor(glm::vec3(r, g, b)); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case MtlElement::D: // opacity | ||||||
|  |         { | ||||||
|  |             float d = p.TakeFloat(); | ||||||
|  |             if (currentMaterial) currentMaterial->SetOpacity(d); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case MtlElement::ILLUM: // illumination model | ||||||
|  |         { | ||||||
|  |             int illum = p.TakeInt(); | ||||||
|  |             if (currentMaterial) currentMaterial->SetIllumination(illum); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case MtlElement::MAP_KD: // diffuse texture map | ||||||
|  |         { | ||||||
|  |             // take rest of line as texture path (can contain spaces) | ||||||
|  |             char* texPath = p.TakeUntil('\0'); | ||||||
|  |             if (texPath && currentMaterial) { | ||||||
|  |                 // trim trailing spaces | ||||||
|  |                 size_t len = std::strlen(texPath); | ||||||
|  |                 while (len > 0 && (texPath[len - 1] == ' ' || texPath[len - 1] == '\t')) | ||||||
|  |                     texPath[--len] = '\0'; | ||||||
|  |  | ||||||
|  |                 currentMaterial->SetDiffuseTexture(Texture::LoadFile(texPath)); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case MtlElement::MAP_KA: // ambient texture map | ||||||
|  |         { | ||||||
|  |             char* texPath = p.TakeUntil('\0'); | ||||||
|  |             if (texPath && currentMaterial) { | ||||||
|  |                 size_t len = std::strlen(texPath); | ||||||
|  |                 while (len > 0 && (texPath[len - 1] == ' ' || texPath[len - 1] == '\t')) | ||||||
|  |                     texPath[--len] = '\0'; | ||||||
|  |  | ||||||
|  |                 // optional: handle ambient texture | ||||||
|  |                 // currentMaterial->SetAmbientTexture(Texture::LoadFile(texPath)); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         default: | ||||||
|  |             // ignore unknown tokens | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Commit last material if pending | ||||||
|  |     if (currentMaterial) { | ||||||
|  |         AddMaterial(currentMaterialName, std::move(currentMaterial)); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     file.close(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Object::AddMaterial(std::string name, std::shared_ptr<Material> material) | ||||||
|  | { | ||||||
|  |     m_materials.insert(std::make_pair(std::move(name), std::move(material))); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | std::shared_ptr<Material> Object::GetMaterial(std::string name) | ||||||
|  | { | ||||||
|  |     auto material = m_materials.find(name); | ||||||
|  |     if (material == m_materials.end()) return nullptr; | ||||||
|  |     return material->second; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Object::CreateNewMesh(const std::string& materialName) | ||||||
|  | { | ||||||
|  |     Mesh mesh; | ||||||
|  |     mesh.materialName = materialName; | ||||||
|  |     m_meshes.push_back(mesh); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Mesh& Object::GetLastMesh() | ||||||
|  | { | ||||||
|  |     if (m_meshes.empty()) { | ||||||
|  |         auto material = std::make_shared<Material>(); | ||||||
|  |         material->SetAmbientColor(glm::vec3(0.52f, 0.52f, 0.52f)); | ||||||
|  |         AddMaterial(DEFAULT_MATERIAL_NAME, std::move(material)); | ||||||
|  |         CreateNewMesh(DEFAULT_MATERIAL_NAME); | ||||||
|  |     } | ||||||
|  |     return m_meshes.back(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Object Object::LoadFile(const std::string& filename) { | ||||||
|  |     std::ifstream file(filename); | ||||||
|  |     if (!file.is_open()) { | ||||||
|  |         std::cerr << "Failed to open OBJ file: " << filename << std::endl; | ||||||
|  |         return {}; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Object obj; | ||||||
|  |     char line[1024]; // static buffer for each line (enough for OBJ lines) | ||||||
|  |  | ||||||
|  |     while (file.getline(line, sizeof(line))) { | ||||||
|  |         Parser p(line); | ||||||
|  |         char* prefix = p.TakeWord(); | ||||||
|  |         if (!prefix) continue; | ||||||
|  |  | ||||||
|  |         switch (toElement(prefix)) { | ||||||
|  |         case ObjElement::OHASH: // comment | ||||||
|  |             continue; | ||||||
|  |  | ||||||
|  |         case ObjElement::MTLLIB: | ||||||
|  |         { | ||||||
|  |             char* mtlFile = p.TakeWord(); | ||||||
|  |             if (mtlFile) { | ||||||
|  |                 std::filesystem::path fullPath = filename; | ||||||
|  |                 std::filesystem::path mtlPath = fullPath.replace_filename(mtlFile); | ||||||
|  |                 obj.LoadMaterials(mtlPath); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case ObjElement::USEMTL: | ||||||
|  |         { | ||||||
|  |             char* materialName = p.TakeWord(); | ||||||
|  |             if (materialName) { | ||||||
|  |                 auto& mesh = obj.GetLastMesh(); | ||||||
|  |                 if (mesh.materialName != materialName) { | ||||||
|  |                     Mesh newMesh; | ||||||
|  |                     newMesh.materialName = materialName; | ||||||
|  |                     obj.m_meshes.push_back(newMesh); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case ObjElement::O: // object name | ||||||
|  |         { | ||||||
|  |             char* name = p.TakeWord(); | ||||||
|  |             if (name) obj.m_name = name; | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case ObjElement::V: // vertex | ||||||
|  |         { | ||||||
|  |             float x = p.TakeFloat(); | ||||||
|  |             float y = p.TakeFloat(); | ||||||
|  |             float z = p.TakeFloat(); | ||||||
|  |             float w = p.TakeFloat(); | ||||||
|  |  | ||||||
|  |             if (w != 0.0f && w != 1.0f) { | ||||||
|  |                 x /= w; y /= w; z /= w; | ||||||
|  |             } | ||||||
|  |             obj.m_vertices.emplace_back(x, y, z); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case ObjElement::VN: // normal | ||||||
|  |         { | ||||||
|  |             float x = p.TakeFloat(); | ||||||
|  |             float y = p.TakeFloat(); | ||||||
|  |             float z = p.TakeFloat(); | ||||||
|  |             obj.m_normals.emplace_back(x, y, z); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case ObjElement::VT: // texcoord | ||||||
|  |         { | ||||||
|  |             float u = p.TakeFloat(); | ||||||
|  |             float v = p.TakeFloat(); | ||||||
|  |             obj.m_texCoords.emplace_back(u, 1.0f - v); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         case ObjElement::F: // face | ||||||
|  |         { | ||||||
|  |             auto& mesh = obj.GetLastMesh(); | ||||||
|  |             int raw_vi, raw_ti, raw_ni; | ||||||
|  |  | ||||||
|  |             while (p.TakeFaceIndices(raw_vi, raw_ti, raw_ni)) { | ||||||
|  |                 // Convert raw OBJ indices to 0-based / -1 sentinel | ||||||
|  |                 int vi = Object::NormalizeIndex(raw_vi, (int)obj.m_vertices.size()); | ||||||
|  |                 int ti = Object::NormalizeIndex(raw_ti, (int)obj.m_texCoords.size()); | ||||||
|  |                 int ni = Object::NormalizeIndex(raw_ni, (int)obj.m_normals.size()); | ||||||
|  |  | ||||||
|  |                 if (vi < 0) { | ||||||
|  |                     // malformed token (no vertex) — skip | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 glm::vec3 vert = obj.m_vertices[vi]; | ||||||
|  |                 glm::vec3 norm(0.0f); | ||||||
|  |                 glm::vec2 texCoord(0.0f); | ||||||
|  |  | ||||||
|  |                 if (ni >= 0) norm = obj.m_normals[ni]; | ||||||
|  |                 if (ti >= 0) texCoord = obj.m_texCoords[ti]; | ||||||
|  |  | ||||||
|  |                 mesh.m_vertexBuffer.emplace_back(vert, norm, texCoord); | ||||||
|  |                 mesh.m_indexBuffer.push_back(mesh.m_vertexBuffer.size() - 1); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         default: | ||||||
|  |             // ignore unknown tokens | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     std::cout << "Object name: " << obj.m_name << std::endl; | ||||||
|  |     std::cout << "Vertices count: " << obj.m_vertices.size() << std::endl; | ||||||
|  |     std::cout << "Normals count: " << obj.m_normals.size() << std::endl; | ||||||
|  |     std::cout << "TexCoords count: " << obj.m_texCoords.size() << std::endl; | ||||||
|  |     std::cout << "Meshes count: " << obj.m_meshes.size() << std::endl; | ||||||
|  |     std::cout << "Materials count: " << obj.m_materials.size() << std::endl; | ||||||
|  |  | ||||||
|  |     file.close(); | ||||||
|  |  | ||||||
|  |     for (auto &mesh : obj.m_meshes) { | ||||||
|  |         mesh.Upload(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return obj; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | void Object::Render(Shader& shader) | ||||||
|  | { | ||||||
|  |     for (auto &mesh : m_meshes) { | ||||||
|  |         auto material = GetMaterial(mesh.materialName); | ||||||
|  |          | ||||||
|  |         shader.setFloat("ambientStrength", 0.2f); | ||||||
|  |         shader.setFloat("shininess", material->GetSpecularWeight()); | ||||||
|  |         shader.setFloat("opacity", material->GetOpacity()); | ||||||
|  |         shader.setBool("useSpecular", material->GetIllumination() >= 2); | ||||||
|  |         shader.setFloat("specularStrength", 1.0f); | ||||||
|  |         shader.setVec3("ambientColor", material->GetAmbientColor()); | ||||||
|  |         shader.setVec3("diffuseColor", material->GetDiffuseColor()); | ||||||
|  |         shader.setVec3("specularColor", material->GetSpecularColor()); | ||||||
|  |  | ||||||
|  |         if (material->HasDiffuseTexture()) { | ||||||
|  |             shader.setBool("useTexture", true); | ||||||
|  |             glActiveTexture(GL_TEXTURE0); | ||||||
|  |             glBindTexture(GL_TEXTURE_2D, material->GetDiffuseTexture()->GetID()); | ||||||
|  |             shader.setInt("diffuseTex", 0); | ||||||
|  |         } else { | ||||||
|  |             shader.setBool("useTexture", false); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         mesh.Render(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
							
								
								
									
										126
									
								
								src/shaders/pbr.fs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										126
									
								
								src/shaders/pbr.fs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,126 @@ | |||||||
|  | #version 410 core | ||||||
|  |  | ||||||
|  | // Output color | ||||||
|  | out vec4 FragColor; | ||||||
|  |  | ||||||
|  | in vec3 vertexPos; | ||||||
|  | in vec3 vertexNormal; | ||||||
|  | in vec2 TexCoords; | ||||||
|  |  | ||||||
|  | uniform vec3 lightPos; | ||||||
|  | uniform vec3 viewPos; | ||||||
|  |  | ||||||
|  | // From Object Renderer | ||||||
|  | uniform vec3 ambientColor; | ||||||
|  | uniform vec3 diffuseColor; | ||||||
|  | uniform vec3 specularColor;   // used as F0 (base reflectance) | ||||||
|  |  | ||||||
|  | uniform float ambientStrength; | ||||||
|  | uniform float specularStrength;  | ||||||
|  | uniform float shininess;      // mapped to roughness | ||||||
|  | uniform bool useSpecular; | ||||||
|  |  | ||||||
|  | uniform float opacity; | ||||||
|  |  | ||||||
|  | uniform sampler2D diffuseTex; | ||||||
|  | uniform bool useTexture; | ||||||
|  |  | ||||||
|  | #define LIGHT_COLOR vec3(1.0, 1.0, 1.0) | ||||||
|  |  | ||||||
|  | // ---------------------------------------------------------------------------- | ||||||
|  | // Helper functions for Cook-Torrance BRDF | ||||||
|  | // ---------------------------------------------------------------------------- | ||||||
|  |  | ||||||
|  | // Normal Distribution Function (GGX/Trowbridge-Reitz) | ||||||
|  | float DistributionGGX(vec3 N, vec3 H, float roughness) | ||||||
|  | { | ||||||
|  |     float a      = roughness * roughness; | ||||||
|  |     float a2     = a * a; | ||||||
|  |     float NdotH  = max(dot(N, H), 0.0); | ||||||
|  |     float NdotH2 = NdotH * NdotH; | ||||||
|  |  | ||||||
|  |     float num   = a2; | ||||||
|  |     float denom = (NdotH2 * (a2 - 1.0) + 1.0); | ||||||
|  |     denom = 3.14159265 * denom * denom; | ||||||
|  |  | ||||||
|  |     return num / denom; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Geometry function (Schlick-GGX) | ||||||
|  | float GeometrySchlickGGX(float NdotV, float roughness) | ||||||
|  | { | ||||||
|  |     float r = (roughness + 1.0); | ||||||
|  |     float k = (r * r) / 8.0; // remapped for direct lighting | ||||||
|  |  | ||||||
|  |     float num   = NdotV; | ||||||
|  |     float denom = NdotV * (1.0 - k) + k; | ||||||
|  |  | ||||||
|  |     return num / denom; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Smith's geometry function | ||||||
|  | float GeometrySmith(vec3 N, vec3 V, vec3 L, float roughness) | ||||||
|  | { | ||||||
|  |     float NdotV = max(dot(N, V), 0.0); | ||||||
|  |     float NdotL = max(dot(N, L), 0.0); | ||||||
|  |     float ggx1 = GeometrySchlickGGX(NdotV, roughness); | ||||||
|  |     float ggx2 = GeometrySchlickGGX(NdotL, roughness); | ||||||
|  |     return ggx1 * ggx2; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Fresnel term (Schlick's approximation) | ||||||
|  | vec3 FresnelSchlick(float cosTheta, vec3 F0) | ||||||
|  | { | ||||||
|  |     return F0 + (1.0 - F0) * pow(1.0 - cosTheta, 5.0); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ---------------------------------------------------------------------------- | ||||||
|  | void main() | ||||||
|  | { | ||||||
|  |     vec3 N = normalize(vertexNormal); | ||||||
|  |     vec3 V = normalize(viewPos - vertexPos); | ||||||
|  |     vec3 L = normalize(lightPos - vertexPos); | ||||||
|  |     vec3 H = normalize(V + L); | ||||||
|  |  | ||||||
|  |     // Texture or uniform color | ||||||
|  |     vec3 albedo = (useTexture) | ||||||
|  |         ? texture(diffuseTex, TexCoords).rgb | ||||||
|  |         : diffuseColor; | ||||||
|  |  | ||||||
|  |     // Map shininess to roughness (inverse relationship) | ||||||
|  |     float roughness = clamp(1.0 - (shininess / 256.0), 0.05, 1.0); | ||||||
|  |  | ||||||
|  |     // Base reflectivity (F0) | ||||||
|  |     vec3 F0 = mix(vec3(0.04), specularColor, specularStrength); | ||||||
|  |  | ||||||
|  |     // Cook-Torrance BRDF | ||||||
|  |     float NDF = DistributionGGX(N, H, roughness); | ||||||
|  |     float G   = GeometrySmith(N, V, L, roughness); | ||||||
|  |     vec3  F   = FresnelSchlick(max(dot(H, V), 0.0), F0); | ||||||
|  |  | ||||||
|  |     vec3 numerator    = NDF * G * F; | ||||||
|  |     float denominator = 4.0 * max(dot(N, V), 0.0) * max(dot(N, L), 0.0) + 0.001; | ||||||
|  |     vec3 specular     = numerator / denominator; | ||||||
|  |  | ||||||
|  |     // Energy conservation | ||||||
|  |     vec3 kS = F; | ||||||
|  |     vec3 kD = vec3(1.0) - kS; | ||||||
|  |     kD *= (useSpecular ? 1.0 : 1.0); // keep diffuse always unless specular is off | ||||||
|  |  | ||||||
|  |     float NdotL = max(dot(N, L), 0.0); | ||||||
|  |  | ||||||
|  |     vec3 diffuse = kD * albedo / 3.14159265; | ||||||
|  |     vec3 radiance = LIGHT_COLOR; | ||||||
|  |  | ||||||
|  |     vec3 Lo = (diffuse + specular) * radiance * NdotL; | ||||||
|  |  | ||||||
|  |     // Ambient (simple, not IBL) | ||||||
|  |     vec3 ambient = ambientStrength * ambientColor * albedo; | ||||||
|  |  | ||||||
|  |     vec3 result = ambient + Lo; | ||||||
|  |  | ||||||
|  |     // Gamma correction | ||||||
|  |     result = pow(result, vec3(1.0/2.2)); | ||||||
|  |  | ||||||
|  |     FragColor = vec4(result, opacity); | ||||||
|  | } | ||||||
| @ -17,7 +17,12 @@ uniform vec3 diffuseColor; | |||||||
| uniform vec3 specularColor; | uniform vec3 specularColor; | ||||||
|  |  | ||||||
| uniform float ambientStrength; | uniform float ambientStrength; | ||||||
|  |  | ||||||
| uniform float specularStrength; | uniform float specularStrength; | ||||||
|  | uniform float shininess; | ||||||
|  | uniform bool useSpecular; | ||||||
|  |  | ||||||
|  | uniform float opacity; | ||||||
|  |  | ||||||
| uniform sampler2D diffuseTex; | uniform sampler2D diffuseTex; | ||||||
| uniform bool useTexture; | uniform bool useTexture; | ||||||
| @ -33,8 +38,13 @@ void main() | |||||||
|     vec3 reflectDir = reflect(-lightDir, norm); |     vec3 reflectDir = reflect(-lightDir, norm); | ||||||
|  |  | ||||||
|     // Phong components |     // Phong components | ||||||
|     float spec = pow(max(dot(viewDir, reflectDir), 0.0), 32); |     // float spec = pow(max(dot(viewDir, reflectDir), 0.0), clamp(shininess, 2, 256)); | ||||||
|     vec3 specular = specularStrength * spec * specularColor;  |     // vec3 specular = (useSpecular) ? specularStrength * spec * specularColor : vec3(0.0); | ||||||
|  |  | ||||||
|  |     // Blinn Phong | ||||||
|  |     vec3 halfDir = normalize(lightDir + viewDir); | ||||||
|  |     float spec = pow(max(dot(norm, halfDir), 0.0), clamp(shininess, 2.0, 256.0)); | ||||||
|  |     vec3 specular = (useSpecular) ? specularStrength * spec * specularColor : vec3(0.0); | ||||||
|  |  | ||||||
|     float diff = max(dot(norm, lightDir), 0.0); |     float diff = max(dot(norm, lightDir), 0.0); | ||||||
|     vec3 diffuse = diff * diffuseColor; |     vec3 diffuse = diff * diffuseColor; | ||||||
| @ -46,5 +56,6 @@ void main() | |||||||
|         : diffuseColor; |         : diffuseColor; | ||||||
|  |  | ||||||
|     vec3 result = (ambient + diffuse + specular) * texColor; |     vec3 result = (ambient + diffuse + specular) * texColor; | ||||||
|     FragColor = vec4(result, 1.0); |  | ||||||
|  |     FragColor = vec4(result, opacity); | ||||||
| } | } | ||||||
							
								
								
									
										151
									
								
								src/vertex.cpp
									
									
									
									
									
								
							
							
						
						
									
										151
									
								
								src/vertex.cpp
									
									
									
									
									
								
							| @ -1,151 +0,0 @@ | |||||||
| #include <GL/glew.h> |  | ||||||
|  |  | ||||||
| #include "vertex.h" |  | ||||||
|  |  | ||||||
| #define BLOCK_SIZE 0.5f |  | ||||||
|  |  | ||||||
| Point::Point(glm::vec3 position, glm::vec3 normal, glm::vec4 color) |  | ||||||
| { |  | ||||||
|     m_position = position; |  | ||||||
|     m_normal = normal; |  | ||||||
|     m_color = color; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| Vertices::Vertices() |  | ||||||
| { |  | ||||||
|     m_items = std::vector<Point>(); |  | ||||||
|  |  | ||||||
|     glGenVertexArrays(1, &m_vao); |  | ||||||
|     glGenBuffers(1, &m_vbo); |  | ||||||
|     glGenBuffers(1, &m_ebo); |  | ||||||
|  |  | ||||||
|     glBindVertexArray(m_vao); |  | ||||||
|  |  | ||||||
|     // VBO (vertex buffer) |  | ||||||
|     glBindBuffer(GL_ARRAY_BUFFER, m_vbo); |  | ||||||
|     glBufferData(GL_ARRAY_BUFFER, 0, nullptr, GL_DYNAMIC_DRAW); |  | ||||||
|  |  | ||||||
|     // EBO (index buffer) |  | ||||||
|     glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_ebo); |  | ||||||
|     glBufferData(GL_ELEMENT_ARRAY_BUFFER, 0, nullptr, GL_DYNAMIC_DRAW); |  | ||||||
|  |  | ||||||
|     // glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_ebo); |  | ||||||
|     // glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(triangleIndices), triangleIndices, GL_STATIC_DRAW); |  | ||||||
|  |  | ||||||
|     glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, sizeof(Point), (void*)0); |  | ||||||
|     glEnableVertexAttribArray(0); |  | ||||||
|  |  | ||||||
|     glVertexAttribPointer(1, 3, GL_FLOAT, GL_FALSE, sizeof(Point), (void*)(3 * sizeof(float))); |  | ||||||
|     glEnableVertexAttribArray(1); |  | ||||||
|      |  | ||||||
|     glVertexAttribPointer(2, 4, GL_FLOAT, GL_FALSE, sizeof(Point), (void*)(6 * sizeof(float))); |  | ||||||
|     glEnableVertexAttribArray(2); |  | ||||||
|  |  | ||||||
|     glBindBuffer(GL_ARRAY_BUFFER, 0); |  | ||||||
|     glBindVertexArray(0); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Call this after you finish adding vertices (or call it each time after PushBlock) |  | ||||||
| void Vertices::Upload() |  | ||||||
| { |  | ||||||
|     glBindVertexArray(m_vao); |  | ||||||
|  |  | ||||||
|     glBindBuffer(GL_ARRAY_BUFFER, m_vbo); |  | ||||||
|     glBufferData(GL_ARRAY_BUFFER, m_items.size() * sizeof(Point), m_items.data(), GL_DYNAMIC_DRAW); |  | ||||||
|  |  | ||||||
|     // Upload indices |  | ||||||
|     glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, m_ebo); |  | ||||||
|     glBufferData(GL_ELEMENT_ARRAY_BUFFER, m_indices.size() * sizeof(unsigned int), m_indices.data(), GL_STATIC_DRAW); |  | ||||||
|  |  | ||||||
|     glBindVertexArray(0); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // void Vertices::PushBlock(const Block& block) |  | ||||||
| // { |  | ||||||
| //     // 1 face |  | ||||||
| //     m_items.emplace_back(block.Position(), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, -BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, -BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 2 face |  | ||||||
| //     m_items.emplace_back(block.Position(), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, 0.f, 0.f), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, -BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 3 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, -BLOCK_SIZE, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, -BLOCK_SIZE, -BLOCK_SIZE), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 4 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, -BLOCK_SIZE, -BLOCK_SIZE), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 5 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position(), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, -BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 6 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, -BLOCK_SIZE, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, -BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 7 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, 0.f, 0.f), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, -BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 8 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, -BLOCK_SIZE, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, -BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 9 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, 0.f, 0.f), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position(), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 10 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, 0.f, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, 0.f, 0.f), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 11 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, BLOCK_SIZE, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
|  |  | ||||||
| //     // 12 face |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(0.f, BLOCK_SIZE, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, BLOCK_SIZE, -BLOCK_SIZE), block.Color()); |  | ||||||
| //     m_items.emplace_back(block.Position() + glm::vec3(BLOCK_SIZE, BLOCK_SIZE, 0.f), block.Color()); |  | ||||||
| // } |  | ||||||
|  |  | ||||||
| void Vertices::PushVertex(const Point& point) |  | ||||||
| { |  | ||||||
|     m_items.push_back(point); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Vertices::PushIndex(unsigned int index) |  | ||||||
| { |  | ||||||
|     m_indices.push_back(index); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Vertices::Bind() |  | ||||||
| { |  | ||||||
|     glBindVertexArray(m_vao); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Vertices::Unbind() |  | ||||||
| { |  | ||||||
|     glBindVertexArray(0); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Vertices::Draw() |  | ||||||
| { |  | ||||||
|     Bind(); |  | ||||||
|     glDrawElements(GL_TRIANGLES, static_cast<GLsizei>(m_indices.size()), GL_UNSIGNED_INT, 0); |  | ||||||
|     Unbind(); |  | ||||||
| } |  | ||||||
		Reference in New Issue
	
	Block a user