added: Textures and fixed ECS

This commit is contained in:
OusmBlueNinja 2024-12-25 00:23:10 -06:00
parent 4d18863e97
commit fede442827
7 changed files with 126 additions and 32 deletions

BIN
assets/textures/default.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 138 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

View File

@ -1,4 +1,4 @@
// ECS.h // src/ECS.h
#pragma once #pragma once
#include <unordered_map> #include <unordered_map>

View File

@ -11,6 +11,14 @@
#include <stdio.h> #include <stdio.h>
#include <vector> #include <vector>
#include <string> #include <string>
#include <unordered_map>
#include <typeindex>
#include <typeinfo>
#include <memory>
#include <algorithm>
#define STB_IMAGE_IMPLEMENTATION
#include "stb_image.h" // Include stb_image.h
#include "ECS.h" // Include the ECS header #include "ECS.h" // Include the ECS header
@ -58,6 +66,74 @@ private:
#define LOGPOINT(msg) #define LOGPOINT(msg)
#endif #endif
// =====================
// TextureManager Implementation
// =====================
class TextureManager {
public:
GLuint LoadTexture(const std::string& path) {
// Check if texture already loaded
auto it = textures.find(path);
if (it != textures.end()) {
return it->second;
}
// Load image
int width, height, channels;
unsigned char* data = stbi_load(path.c_str(), &width, &height, &channels, 4);
if (!data) {
Logger::GetInstance().Log(LogLevel::ERROR, "Failed to load texture: " + path);
return 0;
}
// Generate OpenGL texture
GLuint textureID;
glGenTextures(1, &textureID);
glBindTexture(GL_TEXTURE_2D, textureID);
// Set texture parameters
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
// Upload texture data
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, data);
//glGenerateMipmap(GL_TEXTURE_2D);
glBindTexture(GL_TEXTURE_2D, 0);
// Free image data
stbi_image_free(data);
// Store texture
textures[path] = textureID;
Logger::GetInstance().Log(LogLevel::INFO, "Loaded texture: " + path);
return textureID;
}
GLuint GetTexture(const std::string& path) {
auto it = textures.find(path);
if (it != textures.end()) {
return it->second;
}
return LoadTexture(path);
}
void Cleanup() {
for (auto& pair : textures) {
glDeleteTextures(1, &pair.second);
}
textures.clear();
}
private:
std::unordered_map<std::string, GLuint> textures;
};
// Instantiate TextureManager
TextureManager textureManager;
// ===================== // =====================
// ECS Instances // ECS Instances
// ===================== // =====================
@ -149,18 +225,40 @@ void ShowViewport()
// Get the size of the viewport // Get the size of the viewport
ImVec2 viewport_size = ImGui::GetContentRegionAvail(); ImVec2 viewport_size = ImGui::GetContentRegionAvail();
// For demonstration, we'll render a colored rectangle // Set viewport background to black
// In a real engine, you'd render your scene here ImDrawList* draw_list = ImGui::GetWindowDrawList();
// Calculate the position relative to window
ImVec2 pos = ImGui::GetCursorScreenPos(); ImVec2 pos = ImGui::GetCursorScreenPos();
// Define rectangle dimensions
ImVec2 rect_min = pos; ImVec2 rect_min = pos;
ImVec2 rect_max = ImVec2(pos.x + viewport_size.x, pos.y + viewport_size.y); ImVec2 rect_max = ImVec2(pos.x + viewport_size.x, pos.y + viewport_size.y);
draw_list->AddRectFilled(rect_min, rect_max, IM_COL32(0, 0, 0, 255)); // Black background
// Render a colored rectangle // Iterate through active entities
ImGui::GetWindowDrawList()->AddRectFilled(rect_min, rect_max, IM_COL32(100, 100, 200, 255)); const auto& entities = entityManager.GetActiveEntities();
for (auto entity : entities) {
if (componentManager.HasComponent<SpriteComponent>(entity) && componentManager.HasComponent<TransformComponent>(entity)) {
auto& sprite = componentManager.GetComponent<SpriteComponent>(entity);
auto& transform = componentManager.GetComponent<TransformComponent>(entity);
// Get texture ID
GLuint textureID = textureManager.GetTexture(sprite.texturePath);
if (textureID == 0) {
continue; // Failed to load texture
}
// Convert GLuint to ImTextureID
ImTextureID imgui_tex_id = (ImTextureID)(intptr_t)textureID;
// Define size based on scale
ImVec2 size(transform.scale.x, transform.scale.y);
// Define position based on position
ImVec2 img_pos = ImVec2(pos.x + transform.position.x, pos.y + transform.position.y);
ImVec2 img_size = ImVec2(img_pos.x + size.x, img_pos.y + size.y);
// Render the image
draw_list->AddImage(imgui_tex_id, img_pos, img_size);
}
}
ImGui::End(); ImGui::End();
} }
@ -289,11 +387,15 @@ void ShowInspector(EntityManager& em, ComponentManager& cm, Entity selectedEntit
if (cm.HasComponent<SpriteComponent>(selectedEntity)) { if (cm.HasComponent<SpriteComponent>(selectedEntity)) {
if (ImGui::TreeNode("Sprite")) { if (ImGui::TreeNode("Sprite")) {
auto& sprite = cm.GetComponent<SpriteComponent>(selectedEntity); auto& sprite = cm.GetComponent<SpriteComponent>(selectedEntity);
char buffer[256]; static char buffer[256];
strncpy(buffer, sprite.texturePath.c_str(), sizeof(buffer)); strncpy(buffer, sprite.texturePath.c_str(), sizeof(buffer));
buffer[sizeof(buffer) - 1] = '\0'; // Ensure null-termination buffer[sizeof(buffer) - 1] = '\0'; // Ensure null-termination
if (ImGui::InputText("Texture Path", buffer, sizeof(buffer))) { if (ImGui::InputText("Texture Path", buffer, sizeof(buffer))) {
sprite.texturePath = std::string(buffer); std::string newPath(buffer);
if (newPath != sprite.texturePath) {
sprite.texturePath = newPath;
Logger::GetInstance().Log(LogLevel::INFO, "Updated SpriteComponent texture path to " + newPath + " for Entity " + std::to_string(selectedEntity));
}
} }
if (ImGui::Button("Remove Sprite")) { if (ImGui::Button("Remove Sprite")) {
@ -309,8 +411,11 @@ void ShowInspector(EntityManager& em, ComponentManager& cm, Entity selectedEntit
} }
else { else {
if (ImGui::Button("Add Sprite")) { if (ImGui::Button("Add Sprite")) {
cm.AddComponent<SpriteComponent>(selectedEntity, SpriteComponent("path/to/texture.png")); // Prompt the user to input texture path
Logger::GetInstance().Log(LogLevel::INFO, "Added SpriteComponent to Entity " + std::to_string(selectedEntity)); // For simplicity, assign a default path
std::string defaultPath = "assets/textures/default.png"; // Ensure this texture exists
cm.AddComponent<SpriteComponent>(selectedEntity, SpriteComponent(defaultPath));
Logger::GetInstance().Log(LogLevel::INFO, "Added SpriteComponent to Entity " + std::to_string(selectedEntity) + " with texture path: " + defaultPath);
} }
} }
@ -345,6 +450,7 @@ int main(int, char**)
glfwMakeContextCurrent(window); glfwMakeContextCurrent(window);
glfwSwapInterval(1); // Enable vsync glfwSwapInterval(1); // Enable vsync
// Setup Dear ImGui context // Setup Dear ImGui context
IMGUI_CHECKVERSION(); IMGUI_CHECKVERSION();
ImGui::CreateContext(); ImGui::CreateContext();
@ -408,7 +514,7 @@ int main(int, char**)
componentManager.RegisterComponent<TransformComponent>(); componentManager.RegisterComponent<TransformComponent>();
componentManager.RegisterComponent<SpriteComponent>(); componentManager.RegisterComponent<SpriteComponent>();
// Create a default entity with TransformComponent // Create a default entity with TransformComponent (without SpriteComponent)
Entity defaultEntity = entityManager.CreateEntity(); Entity defaultEntity = entityManager.CreateEntity();
componentManager.AddComponent<TransformComponent>(defaultEntity, TransformComponent()); componentManager.AddComponent<TransformComponent>(defaultEntity, TransformComponent());
@ -436,8 +542,10 @@ int main(int, char**)
ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoMove |
ImGuiWindowFlags_NoBringToFrontOnFocus | ImGuiWindowFlags_NoNavFocus; ImGuiWindowFlags_NoBringToFrontOnFocus | ImGuiWindowFlags_NoNavFocus;
const ImGuiViewport* viewport = ImGui::GetMainViewport(); const ImGuiViewport* viewport = ImGui::GetMainViewport();
ImGui::SetNextWindowPos(viewport->Pos);
ImGui::SetNextWindowSize(viewport->Size); // Adjust the docking space position by 20 pixels to avoid overlapping with the main menu bar
ImGui::SetNextWindowPos(ImVec2(viewport->Pos.x, viewport->Pos.y + 20));
ImGui::SetNextWindowSize(ImVec2(viewport->Size.x, viewport->Size.y - 20));
ImGui::SetNextWindowViewport(viewport->ID); ImGui::SetNextWindowViewport(viewport->ID);
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0f); ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0f);
ImGui::PushStyleVar(ImGuiStyleVar_WindowBorderSize, 0.0f); ImGui::PushStyleVar(ImGuiStyleVar_WindowBorderSize, 0.0f);
@ -486,6 +594,7 @@ int main(int, char**)
} }
// Cleanup // Cleanup
textureManager.Cleanup();
ImGui_ImplOpenGL3_Shutdown(); ImGui_ImplOpenGL3_Shutdown();
ImGui_ImplGlfw_Shutdown(); ImGui_ImplGlfw_Shutdown();
ImGui::DestroyContext(); ImGui::DestroyContext();

View File

@ -1,11 +0,0 @@
cmake_minimum_required(VERSION 3.22)
project(stb)
file(GLOB_RECURSE SOURCES src/*.cpp)
add_library(stb)
target_include_directories(stb PUBLIC include)
target_sources(stb PRIVATE ${SOURCES})

View File

@ -1,5 +1,4 @@
#ifndef STB_IMAG_H #pragma once
#define STB_IMAG_H
/* stb_image - v2.30 - public domain image loader - http://nothings.org/stb /* stb_image - v2.30 - public domain image loader - http://nothings.org/stb
no warranty implied; use at your own risk no warranty implied; use at your own risk
@ -7990,4 +7989,3 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
------------------------------------------------------------------------------ ------------------------------------------------------------------------------
*/ */
#endif

View File

@ -1,2 +0,0 @@
#define STB_IMAGE_IMPLEMENTATION
#include <stb_image.h>