diff --git a/.gitignore b/.gitignore index b76cd25..2ad32c4 100644 --- a/.gitignore +++ b/.gitignore @@ -2,4 +2,4 @@ [Oo]ut/ [Bb]uild/ -*.json \ No newline at end of file +/*.json \ No newline at end of file diff --git a/CMakeLists.txt b/CMakeLists.txt index 1fd9ed9..63c3e6a 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -5,8 +5,8 @@ cmake_minimum_required (VERSION 3.8) project ("KoiKoi") - # Include sub-projects. add_subdirectory ("vendor/spdlog") add_subdirectory ("vendor/glfw") +add_subdirectory ("vendor/lol") add_subdirectory ("src") diff --git a/res/card.fs b/res/card.fs new file mode 100644 index 0000000..d37fd40 --- /dev/null +++ b/res/card.fs @@ -0,0 +1,12 @@ +#version 440 core + +in vec2 uvCoord; +out vec4 FragColor; + +uniform vec2 offset; +uniform sampler2D motive; + +void main() +{ + FragColor = texture(motive, uvCoord + offset); +} \ No newline at end of file diff --git a/res/card.vs b/res/card.vs new file mode 100644 index 0000000..e7367fa --- /dev/null +++ b/res/card.vs @@ -0,0 +1,15 @@ +#version 440 core + +layout (location = 0) in vec2 position; +layout (location = 1) in vec2 UV; + +out vec2 uvCoord; + +uniform mat4 view; +uniform mat4 projection; + +void main() +{ + uvCoord = UV; + gl_Position = projection * view * vec4(position, 0.0f, 1.0f); +} \ No newline at end of file diff --git a/res/card_atlas.json b/res/card_atlas.json new file mode 100644 index 0000000..2c731c8 --- /dev/null +++ b/res/card_atlas.json @@ -0,0 +1,34 @@ +[ + { + "posX": 0, + "posY": 0, + "month": 0, + "type": 0, + "name_en": "Crane and Sun", + "name_jp": "松に鶴" + }, + { + "posX": 0, + "posY": 1, + "month": 0, + "type": 2, + "name_en": "Poetry ribbon", + "name_jp": "松に赤短" + }, + { + "posX": 0, + "posY": 2, + "month": 0, + "type": 3, + "name_en": "Pine plain", + "name_jp": "松のカス" + }, + { + "posX": 0, + "posY": 3, + "month": 0, + "type": 3, + "name_en": "Pine plain", + "name_jp": "松のカス" + } +] \ No newline at end of file diff --git a/res/card_atlas.png b/res/card_atlas.png new file mode 100644 index 0000000..26645a8 Binary files /dev/null and b/res/card_atlas.png differ diff --git a/src/Application.cpp b/src/Application.cpp index 54e22bc..d52aad2 100644 --- a/src/Application.cpp +++ b/src/Application.cpp @@ -1,9 +1,17 @@ #include "Application.hpp" +#include + +#include #include #include +#include + #include "Window.hpp" +#include "Card.hpp" + +#include "resources.h" void Application::Run() { @@ -14,9 +22,32 @@ void Application::Run() } spdlog::debug("Launched Application"); + + int currentMonth = 0; + int currentType = 0; + std::chrono::system_clock::time_point begin = std::chrono::system_clock::now(); while (!window->ShouldClose()) { glfwPollEvents(); + + if (std::chrono::duration_cast(std::chrono::system_clock::now() - begin).count() > 250) + { + begin = std::chrono::system_clock::now(); + currentType++; + if (currentType > 3) + { + currentMonth++; + if (currentMonth > 11) + currentMonth = 0; + + currentType = 0; + } + + card->UpdateSuitAndType(static_cast(currentMonth), currentType); + } + + window->Clear(); + window->Draw(*card); window->Display(); } } @@ -36,12 +67,32 @@ Application::Application() : } spdlog::debug("GLFW initialized"); - window = new Window(800, 800, "Koi Koi"); + window = new Window(1280, 720, "Koi Koi"); valid = window->IsValid(); + + if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress)) + { + spdlog::critical("Failed to load GL loader"); + valid = false; + return; + } + spdlog::debug("Loaded GL loader"); + + spdlog::debug("Loading resources..."); + + lol::Image image((unsigned char*)card_atlas_png, card_atlas_png_size); + manager.Create(0, image, lol::TextureFormat::RGBA); + manager.Create(1, std::string((char*)card_vs, card_vs_size), std::string((char*)card_fs, card_fs_size)); + spdlog::debug("Done!"); + + card = new Card(manager, Month::August, 0); + + glViewport(0, 0, 1280, 720); } Application::~Application() { + delete card; delete window; glfwTerminate(); diff --git a/src/Application.hpp b/src/Application.hpp index 270a0b1..523bf14 100644 --- a/src/Application.hpp +++ b/src/Application.hpp @@ -1,6 +1,9 @@ #pragma once +#include + class Window; +class Card; class Application { @@ -18,6 +21,9 @@ private: bool valid; Window* window; + lol::ObjectManager manager; + Card* card; + private: Application(); ~Application(); diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 5375c2d..891456d 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -2,14 +2,46 @@ add_executable(koikoi "main.cpp" "Application.cpp" "Window.cpp" - ) + "Card.cpp") -target_include_directories(koikoi PRIVATE +set( RC_DEPENDS "" ) + +set(output ${CMAKE_BINARY_DIR}/CMakeFiles/koikoi.dir/autogen/resources.h) +set(dir ${CMAKE_SOURCE_DIR}/res) + +# Create empty output file +file(WRITE ${output} "") +# Collect input files +file(GLOB bins ${dir}/*) +# Iterate through input files +foreach(bin ${bins}) + # Get short filename + string(REGEX MATCH "([^/]+)$" filename ${bin}) + # Replace filename spaces & extension separator for C compatibility + string(REGEX REPLACE "\\.| |-" "_" filename ${filename}) + # Read hex data from file + file(READ ${bin} filedata HEX) + # Convert hex data for C compatibility + string(REGEX REPLACE "([0-9a-f][0-9a-f])" "0x\\1," filedata ${filedata}) + # Append data to output file + file(APPEND ${output} "const unsigned char ${filename}[] = {${filedata}};\nconst unsigned ${filename}_size = sizeof(${filename});\n") +endforeach() +set( RC_DEPENDS ${RC_DEPENDS} ${output} PARENT_SCOPE ) + +add_custom_target( rc ALL DEPENDS ${RC_DEPENDS} ) + +add_dependencies(koikoi rc) + +target_include_directories(koikoi PUBLIC + ${CMAKE_SOURCE_DIR}/vendor/headeronly + ${CMAKE_BINARY_DIR}/CMakeFiles/koikoi.dir/autogen spdlog glfw + lol ) -target_link_libraries(koikoi PRIVATE +target_link_libraries(koikoi spdlog glfw + lol ) \ No newline at end of file diff --git a/src/Card.cpp b/src/Card.cpp new file mode 100644 index 0000000..8ed2b97 --- /dev/null +++ b/src/Card.cpp @@ -0,0 +1,53 @@ +#include "Card.hpp" + +Card::Card(lol::ObjectManager& manager, Month month, int type) +{ + try + { + vao = manager.Get(2); + } + catch (const lol::ObjectNotFoundException& err) + { + std::shared_ptr vbo = std::make_shared(std::vector{ + 0.0f, 0.0f, 0.0f, 1.0f / 4.0f, + 0.0f, 1.0f, 0.0f, 0.0f, + 0.61035f, 1.0f, 1.0f / 12.0f, 0.0f, + 0.61035f, 0.0f, 1.0f / 12.0f, 1.0f / 4.0f + }); + + lol::BufferLayout layout({ + lol::VertexAttribute(lol::Type::Float, 2, false), + lol::VertexAttribute(lol::Type::Float, 2, false) + }); + vbo->SetLayout(layout); + + std::shared_ptr ebo = std::make_shared(std::vector{ 0, 1, 2, 0, 2, 3 }); + + vao = manager.Create(2, vbo, ebo); + } + + shader = manager.Get(1); + cards = manager.Get(0); + + UpdateSuitAndType(month, type); +} + +void Card::UpdateSuitAndType(Month month, int type) +{ + this->suit = month; + this->type = type; + + textureOffset = glm::vec2( + static_cast(this->suit) / 12.0f, + this->type / 4.0f + ); +} + +void Card::PreRender(const lol::CameraBase& camera) +{ + cards->Bind(); + shader->SetUniform("offset", textureOffset); + + shader->SetUniform("view", camera.GetView()); + shader->SetUniform("projection", camera.GetProjection()); +} diff --git a/src/Card.hpp b/src/Card.hpp new file mode 100644 index 0000000..b72827d --- /dev/null +++ b/src/Card.hpp @@ -0,0 +1,26 @@ +#pragma once + +#include + +enum class Month +{ + January, February, March, April, May, June, July, August, September, October, November, December +}; + +class Card : public lol::Drawable, public lol::Transformable +{ +public: + Card(lol::ObjectManager& manager, Month month, int type); + + void UpdateSuitAndType(Month month, int type); + +private: + void PreRender(const lol::CameraBase& camera) override; + +private: + std::shared_ptr cards; + + Month suit; + int type; + glm::vec2 textureOffset; +}; \ No newline at end of file diff --git a/src/Window.cpp b/src/Window.cpp index 4cbde7a..8df0e55 100644 --- a/src/Window.cpp +++ b/src/Window.cpp @@ -1,6 +1,5 @@ #include "Window.hpp" -#include #include Window::Window(int width, int height, const std::string& title) : @@ -19,6 +18,21 @@ Window::Window(int width, int height, const std::string& title) : spdlog::debug("Created GLFWwindow \n\tDimensions = ({}px, {}px), \n\tTitle = {}, \n\tFullscreen = {}", width, height, title, "NO"); glfwMakeContextCurrent(window); + + glfwSetFramebufferSizeCallback(window, + [](GLFWwindow* window, int width, int height) + { + glViewport(0, 0, width, height); + UserData* data = (UserData*)glfwGetWindowUserPointer(window); + data->camera->Update(0.0f, ((float)width / (float)height) * 1.0f, 0.0f, 1.0f, -100.0f, 100.0f); + } + ); + + camera = lol::OrthogonalCamera(0.0f, ((float)width / (float)height) * 1.0f, 0.0f, 1.0f); + data.camera = &camera; + spdlog::debug("Created orthogonal camera"); + + glfwSetWindowUserPointer(window, (void*)&data); } Window::~Window() @@ -30,6 +44,16 @@ Window::~Window() } } +void Window::Clear() +{ + glClear(GL_COLOR_BUFFER_BIT); +} + +void Window::Draw(lol::Drawable& drawable) +{ + camera.Draw(drawable); +} + void Window::Display() { glfwSwapBuffers(window); diff --git a/src/Window.hpp b/src/Window.hpp index 15ba771..7378653 100644 --- a/src/Window.hpp +++ b/src/Window.hpp @@ -1,7 +1,14 @@ #pragma once #include +#include #include +#include + +struct UserData +{ + lol::OrthogonalCamera* camera; +}; class Window { @@ -12,9 +19,14 @@ public: inline bool IsValid() { return valid; } inline bool ShouldClose() { return glfwWindowShouldClose(window); } + void Clear(); + void Draw(lol::Drawable& drawable); void Display(); private: bool valid; GLFWwindow* window; + UserData data; + + lol::OrthogonalCamera camera; }; diff --git a/vendor/lol b/vendor/lol index b702bd0..7b268ef 160000 --- a/vendor/lol +++ b/vendor/lol @@ -1 +1 @@ -Subproject commit b702bd00f4e61e7ca1a6fd3da78de87c730e7503 +Subproject commit 7b268efa6a4a893e64c2cad18a05aa633b5aebe1