templatize code; deserialize of sets coming
This commit is contained in:
parent
c05edabf0b
commit
9926c3b617
@ -2,5 +2,7 @@ cmake_minimum_required(VERSION 3.23)
|
||||
|
||||
project(MHache VERSION 0.1)
|
||||
|
||||
set (CMAKE_CXX_STANDARD 17)
|
||||
|
||||
add_subdirectory(dependencies)
|
||||
add_subdirectory(src)
|
||||
|
BIN
fonts/Sweet16mono.ttf
Normal file
BIN
fonts/Sweet16mono.ttf
Normal file
Binary file not shown.
@ -4,50 +4,123 @@
|
||||
#include <string>
|
||||
#include <array>
|
||||
#include <vector>
|
||||
#include <memory>
|
||||
#include <set>
|
||||
#include <functional>
|
||||
#include <map>
|
||||
|
||||
#include "mhache_skill.h"
|
||||
|
||||
class MHacheSkill;
|
||||
|
||||
template <class P>
|
||||
struct PtrLessComparator {
|
||||
bool operator () (const P* a, const P* b) const
|
||||
{
|
||||
if (!a) return true;
|
||||
if (!b) return false;
|
||||
return *a < *b;
|
||||
}
|
||||
};
|
||||
|
||||
class MHacheArmor {
|
||||
private:
|
||||
struct Skill {
|
||||
MHacheSkill* _skill;
|
||||
unsigned char _level;
|
||||
public:
|
||||
typedef enum: unsigned char {
|
||||
FIRE,
|
||||
WATER,
|
||||
THUNDER,
|
||||
ICE,
|
||||
DRAGON,
|
||||
count
|
||||
} ResistanceElement;
|
||||
|
||||
class Part {
|
||||
public:
|
||||
typedef enum : unsigned char {
|
||||
HEAD,
|
||||
CHEST,
|
||||
ARMS,
|
||||
WAIST,
|
||||
LEGS,
|
||||
count
|
||||
} Appellation;
|
||||
|
||||
private:
|
||||
friend class MHacheArmor;
|
||||
|
||||
Appellation _label;
|
||||
|
||||
std::string _name;
|
||||
std::vector<unsigned char> _slots;
|
||||
|
||||
std::map<const MHacheSkill*, unsigned char,
|
||||
PtrLessComparator<MHacheSkill>> _skills;
|
||||
|
||||
const MHacheArmor* _armor;
|
||||
|
||||
Part() = default;
|
||||
|
||||
public:
|
||||
~Part() = default;
|
||||
|
||||
const std::string& name() const { return _name; }
|
||||
const std::vector<unsigned char> slots() const { return _slots; }
|
||||
|
||||
// const std::set<Skill> skills() const { return _skills; }
|
||||
|
||||
Part& set_name(std::string&& name)
|
||||
{
|
||||
_name = std::forward<std::string>(name);
|
||||
return *this;
|
||||
}
|
||||
|
||||
Part& set_slots(std::vector<unsigned char>&& slots)
|
||||
{
|
||||
_slots = std::forward<std::vector<unsigned char>>(slots);
|
||||
return *this;
|
||||
}
|
||||
|
||||
Part& add_skill(const MHacheSkill* skill, unsigned char level)
|
||||
{
|
||||
if (skill && level > 0)
|
||||
_skills.insert_or_assign(skill, level);
|
||||
|
||||
return *this;
|
||||
}
|
||||
};
|
||||
|
||||
struct Piece {
|
||||
std::string _name;
|
||||
std::vector<unsigned char> _slots;
|
||||
std::vector<Skill> _skills;
|
||||
MHacheArmor* _armor;
|
||||
};
|
||||
|
||||
std::string _name;
|
||||
std::array<char, 5> _resistances;
|
||||
|
||||
Piece _head;
|
||||
Piece _chest;
|
||||
Piece _arms;
|
||||
Piece _waist;
|
||||
Piece _legs;
|
||||
|
||||
private:
|
||||
MHacheArmor() = default;
|
||||
|
||||
std::string _name;
|
||||
unsigned char _rarity;
|
||||
unsigned short _defense;
|
||||
std::array<char, ResistanceElement::count> _resistances;
|
||||
std::array<Part, Part::count> _parts;
|
||||
|
||||
private:
|
||||
MHacheArmor() = delete;
|
||||
MHacheArmor(const MHacheArmor&) = delete;
|
||||
MHacheArmor(MHacheArmor&&) = delete;
|
||||
MHacheArmor& operator=(const MHacheArmor&) = delete;
|
||||
MHacheArmor& operator=(MHacheArmor&&) = delete;
|
||||
|
||||
public:
|
||||
~MHacheArmor() = default;
|
||||
MHacheArmor(const std::string& name, unsigned char rarity, unsigned short defense, std::array<char, ResistanceElement::count> resistances)
|
||||
: _name(name), _rarity(rarity), _defense(defense), _resistances(resistances), _parts({})
|
||||
{}
|
||||
|
||||
~MHacheArmor() = default;
|
||||
|
||||
bool operator < (const MHacheArmor&) const;
|
||||
bool operator == (const MHacheArmor&) const;
|
||||
bool operator < (const MHacheArmor& armor) const { return _name < armor._name; }
|
||||
bool operator == (const MHacheArmor& armor) const { return _name == armor._name; }
|
||||
|
||||
const std::string& name() const;
|
||||
const std::string& name() const { return _name; }
|
||||
|
||||
const Part& part(Part::Appellation appellation) const { return _parts[appellation]; }
|
||||
|
||||
Part& forge_part(Part::Appellation appellation)
|
||||
{
|
||||
_parts[appellation]._armor = this;
|
||||
return _parts[appellation];
|
||||
}
|
||||
};
|
||||
|
||||
template<>
|
||||
@ -59,4 +132,15 @@ struct std::hash<MHacheArmor>
|
||||
}
|
||||
};
|
||||
|
||||
/*
|
||||
namespace std {
|
||||
template<>
|
||||
struct less<>
|
||||
{
|
||||
bool operator ()(const MHacheArmor::Part::Skill& a, const MHacheArmor::Part::Skill& b) const { return a < b; }
|
||||
bool operator ()(const MHacheArmor::Part::Skill& a, const std::string& b) const { return a < b; }
|
||||
bool operator ()(const std::string& b, const MHacheArmor::Part::Skill& a) const { return a < b; }
|
||||
};
|
||||
}*/
|
||||
|
||||
#endif
|
||||
|
@ -26,9 +26,9 @@ private:
|
||||
static std::unordered_map<std::string, MHacheColor> _colors;
|
||||
|
||||
public:
|
||||
MHacheColor(int r, int g, int b, int a = 255);
|
||||
MHacheColor(unsigned char r, unsigned char g, unsigned char b, unsigned char a = 255u);
|
||||
MHacheColor(float r, float g, float b, float a = 1.0f);
|
||||
constexpr MHacheColor(unsigned char r, unsigned char g, unsigned char b, unsigned char a = 255u);
|
||||
constexpr MHacheColor(int r, int g, int b, int a = 255);
|
||||
constexpr MHacheColor(float r, float g, float b, float a = 1.0f);
|
||||
|
||||
MHacheColor(const MHacheColor&) = default;
|
||||
MHacheColor(MHacheColor&&) = default;
|
||||
|
@ -7,13 +7,67 @@
|
||||
#include <toml++/toml.h>
|
||||
|
||||
#include "mhache_skill.h"
|
||||
#include "mhache_armor.h"
|
||||
|
||||
class MHacheFactory {
|
||||
private:
|
||||
std::set<MHacheSkill> _skills;
|
||||
std::set<MHacheArmor> _armors;
|
||||
|
||||
MHacheFactory() = default;
|
||||
|
||||
toml::table parse_root_table(const std::filesystem::path&);
|
||||
bool parse_root_table(const std::filesystem::path&, toml::table&);
|
||||
|
||||
template <typename T>
|
||||
static bool deserialize_table_node(const toml::table& table, std::string_view node_name, T& node_value)
|
||||
{
|
||||
const auto node = table.get_as<T>(node_name);
|
||||
|
||||
if (node) {
|
||||
node_value = node->template ref<T>();
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
static bool deserialize_table_node(const toml::table& table, std::string_view node_name,
|
||||
std::vector<T>& listof_values)
|
||||
{
|
||||
const auto array = table.get_as<toml::array>(node_name);
|
||||
|
||||
if (array) {
|
||||
listof_values.reserve(array->size());
|
||||
|
||||
for (const auto& entry: *array) {
|
||||
const auto wrap = entry.as<T>();
|
||||
|
||||
if (wrap)
|
||||
listof_values.push_back(wrap->template ref<T>());
|
||||
}
|
||||
}
|
||||
|
||||
return !(listof_values.empty());
|
||||
}
|
||||
|
||||
template <typename T, size_t Length>
|
||||
static bool deserialize_table_node(const toml::table& table, std::string_view node_name,
|
||||
std::array<T, Length>& listof_values)
|
||||
{
|
||||
const auto array = table.get_as<toml::array>(node_name);
|
||||
|
||||
if (array && array->size() == Length) {
|
||||
for (size_t i = 0; i < Length; ++i) {
|
||||
const auto wrap = array->get_as<T>(i);
|
||||
|
||||
if (wrap)
|
||||
listof_values[i] = wrap->template ref<T>();
|
||||
}
|
||||
}
|
||||
|
||||
return !(listof_values.empty());
|
||||
}
|
||||
|
||||
public:
|
||||
~MHacheFactory() = default;
|
||||
@ -21,6 +75,7 @@ public:
|
||||
static MHacheFactory& singleton();
|
||||
|
||||
bool load_skills(const std::filesystem::path&);
|
||||
bool load_armors(const std::filesystem::path&);
|
||||
|
||||
inline const std::set<MHacheSkill>& available_skills() const { return _skills; }
|
||||
};
|
||||
|
@ -21,7 +21,6 @@ private:
|
||||
MHacheSkill& operator=(MHacheSkill&&) = delete;
|
||||
|
||||
public:
|
||||
|
||||
inline
|
||||
MHacheSkill(const std::string& name, const std::string& description, const MHacheColor* color, const std::vector<std::string>& levels)
|
||||
: _name(name), _description(description), _color(color), _levels(levels)
|
||||
|
@ -3,19 +3,15 @@ find_package(OpenGL REQUIRED)
|
||||
add_executable(
|
||||
${PROJECT_NAME}
|
||||
main.cpp
|
||||
# mhache_object.cpp
|
||||
mhache_color.cpp
|
||||
mhache_skill.cpp
|
||||
mhache_armor.cpp
|
||||
mhache_factory.cpp
|
||||
# mhache_tomltools.cpp
|
||||
# mhache_researcher.cpp
|
||||
mhache_window.cpp
|
||||
)
|
||||
|
||||
target_compile_options(
|
||||
${PROJECT_NAME} PUBLIC
|
||||
-std=c++17
|
||||
-Wall
|
||||
-Wextra
|
||||
-pedantic-errors
|
||||
|
@ -1,19 +1 @@
|
||||
#include "mhache_armor.h"
|
||||
|
||||
bool
|
||||
MHacheArmor::operator < (const MHacheArmor& armor) const
|
||||
{
|
||||
return _name < armor._name;
|
||||
}
|
||||
|
||||
bool
|
||||
MHacheArmor::operator == (const MHacheArmor& armor) const
|
||||
{
|
||||
return _name == armor._name;
|
||||
}
|
||||
|
||||
const std::string&
|
||||
MHacheArmor::name() const
|
||||
{
|
||||
return _name;
|
||||
}
|
||||
|
@ -1,7 +1,11 @@
|
||||
#include "mhache_color.h"
|
||||
#include <unordered_map>
|
||||
|
||||
MHacheColor::MHacheColor(int r, int g, int b, int a /* = 255 */)
|
||||
constexpr MHacheColor::MHacheColor(unsigned char r, unsigned char g, unsigned char b, unsigned char a)
|
||||
: _rgba({ r, g, b, a })
|
||||
{}
|
||||
|
||||
constexpr MHacheColor::MHacheColor(int r, int g, int b, int a /* = 255 */)
|
||||
: MHacheColor(
|
||||
static_cast<unsigned char>(r),
|
||||
static_cast<unsigned char>(g),
|
||||
@ -9,11 +13,7 @@ MHacheColor::MHacheColor(int r, int g, int b, int a /* = 255 */)
|
||||
static_cast<unsigned char>(a))
|
||||
{}
|
||||
|
||||
MHacheColor::MHacheColor(unsigned char r, unsigned char g, unsigned char b, unsigned char a)
|
||||
: _rgba({ r, g, b, a })
|
||||
{}
|
||||
|
||||
MHacheColor::MHacheColor(float r, float g, float b, float a)
|
||||
constexpr MHacheColor::MHacheColor(float r, float g, float b, float a)
|
||||
: MHacheColor(touccolor(r), touccolor(g), touccolor(b), touccolor(a))
|
||||
{}
|
||||
|
||||
|
@ -12,65 +12,125 @@ MHacheFactory::singleton()
|
||||
return factory;
|
||||
}
|
||||
|
||||
toml::table
|
||||
MHacheFactory::parse_root_table(const std::filesystem::path& tomlfile)
|
||||
bool
|
||||
MHacheFactory::parse_root_table(const std::filesystem::path& tomlfile, toml::table& root)
|
||||
{
|
||||
auto result = toml::parse_file(tomlfile.u8string());
|
||||
|
||||
if (!result) {
|
||||
std::cerr << "Parsing of TOML file << " << tomlfile << " failed." << std::endl;
|
||||
std::cerr << "-> " << result.error().description() << std::endl;
|
||||
return false;
|
||||
}
|
||||
|
||||
return result.table();
|
||||
root = std::move(result.table());
|
||||
return true;
|
||||
}
|
||||
|
||||
bool
|
||||
MHacheFactory::load_skills(const std::filesystem::path& tomlfile)
|
||||
{
|
||||
const toml::table root = parse_root_table(tomlfile);
|
||||
if (root.empty()) return false;
|
||||
toml::table skills;
|
||||
{
|
||||
toml::table root;
|
||||
if (!parse_root_table(tomlfile, root) || root.empty())
|
||||
return false;
|
||||
|
||||
const toml::table* skills = root["skills"].as_table();
|
||||
if (!skills) return false;
|
||||
if (!deserialize_table_node<toml::table>(root, "skills", skills))
|
||||
return false;
|
||||
}
|
||||
|
||||
for (const auto& sentry: *skills) {
|
||||
const auto stable = sentry.second.as_table();
|
||||
if (!stable) continue;
|
||||
|
||||
const std::string name = std::string(sentry.first.str());
|
||||
if (name.empty()) continue;
|
||||
for (const auto& skill_entry: skills) {
|
||||
const auto p_skill = skill_entry.second.as_table();
|
||||
if (!p_skill) continue;
|
||||
|
||||
const auto& skill = *p_skill;
|
||||
|
||||
MHacheColor* color = nullptr;
|
||||
{
|
||||
const auto scolor = stable->get_as<std::string>("color");
|
||||
if (scolor) color = MHacheColor::get_color(scolor->get());
|
||||
std::string skill_color;
|
||||
if (!deserialize_table_node(skill, "color", skill_color))
|
||||
continue;
|
||||
|
||||
color = MHacheColor::get_color(skill_color);
|
||||
if (!color) continue;
|
||||
}
|
||||
|
||||
std::string description;
|
||||
{
|
||||
const auto sdescription = stable->get_as<std::string>("description");
|
||||
if (!sdescription) continue;
|
||||
description = sdescription->get();
|
||||
}
|
||||
if (!deserialize_table_node(skill, "description", description) || description.empty())
|
||||
continue;
|
||||
|
||||
std::vector<std::string> levels;
|
||||
{
|
||||
const auto slevels = stable->get_as<toml::array>("levels");
|
||||
if (!slevels) continue;
|
||||
if (!deserialize_table_node(skill, "levels", levels) || levels.empty())
|
||||
continue;
|
||||
|
||||
levels.reserve(slevels->size());
|
||||
std::string name = std::string(skill_entry.first.str());
|
||||
if (name.empty()) continue;
|
||||
|
||||
for (const auto& slevel: *slevels) {
|
||||
if (!slevel.is_string()) continue;
|
||||
levels.push_back(slevel.as_string()->get());
|
||||
}
|
||||
|
||||
if (levels.empty()) continue;
|
||||
}
|
||||
|
||||
_skills.emplace(name, description, color, levels);
|
||||
_skills.emplace(std::move(name), std::move(description), color, std::move(levels));
|
||||
}
|
||||
|
||||
return !_skills.empty();
|
||||
return !(_skills.empty());
|
||||
}
|
||||
|
||||
bool
|
||||
MHacheFactory::load_armors(const std::filesystem::path& tomlfile)
|
||||
{
|
||||
toml::table sets;
|
||||
{
|
||||
toml::table root;
|
||||
if (!parse_root_table(tomlfile, root) || root.empty())
|
||||
return false;
|
||||
|
||||
if (!deserialize_table_node(root, "sets", sets))
|
||||
return false;
|
||||
}
|
||||
|
||||
for (const auto& set_entry: sets) {
|
||||
const auto p_set = set_entry.second.as_table();
|
||||
if (!p_set) continue;
|
||||
|
||||
const auto& set = *p_set;
|
||||
|
||||
const std::string name = std::string(set_entry.first.str());
|
||||
if (name.empty()) continue;
|
||||
|
||||
int64_t rarity;
|
||||
if (!deserialize_table_node(set, "rarity", rarity))
|
||||
return false;
|
||||
|
||||
int64_t defense;
|
||||
if (!deserialize_table_node(set, "defense", defense))
|
||||
return false;
|
||||
|
||||
std::array<int64_t, MHacheArmor::ResistanceElement::count> resistances;
|
||||
if (!deserialize_table_node(set, "resistances", resistances))
|
||||
return false;
|
||||
|
||||
MHacheArmor armor(name, rarity, defense, resistances);
|
||||
|
||||
static constexpr std::array<std::string_view, MHacheArmor::Part::Appellation::count> appellations =
|
||||
{
|
||||
"head", "chest", "arms", "waist", "legs"
|
||||
};
|
||||
|
||||
for (auto appellation: appellations)
|
||||
{
|
||||
toml::table part;
|
||||
if (!deserialize_table_node<toml::table>(set, appellation, part))
|
||||
return false;
|
||||
|
||||
|
||||
}
|
||||
|
||||
// MHacheArmor armor()
|
||||
|
||||
|
||||
/*
|
||||
toml::table head;
|
||||
deserialize_table_node(set, "head", head);
|
||||
*/
|
||||
}
|
||||
|
||||
return !_armors.empty();
|
||||
}
|
||||
|
@ -3,7 +3,7 @@
|
||||
#include <array>
|
||||
#include <csignal>
|
||||
#include <iostream>
|
||||
#include <functional>
|
||||
#include <filesystem>
|
||||
|
||||
#include "imgui.h"
|
||||
#include "imgui_impl_glfw.h"
|
||||
@ -22,9 +22,11 @@ MHacheWindow::MHacheWindow(int width, int height)
|
||||
glfwSetErrorCallback(&MHacheWindow::callback_glfw_error);
|
||||
|
||||
if (GLFW_TRUE == glfwInit()) {
|
||||
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, _opengl_major_version);
|
||||
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, _opengl_minor_version);
|
||||
glfwWindowHint(GLFW_RESIZABLE, GLFW_FALSE);
|
||||
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, _opengl_major_version);
|
||||
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, _opengl_minor_version);
|
||||
glfwWindowHint(GLFW_SCALE_TO_MONITOR, GLFW_TRUE);
|
||||
glfwWindowHint(GLFW_RESIZABLE, GLFW_FALSE);
|
||||
glfwWindowHint(GLFW_DECORATED, GLFW_TRUE);
|
||||
|
||||
_window = glfwCreateWindow(_width, _height, _title, nullptr, nullptr);
|
||||
|
||||
@ -41,6 +43,13 @@ MHacheWindow::MHacheWindow(int width, int height)
|
||||
ImGui::CreateContext();
|
||||
ImGui_ImplGlfw_InitForOpenGL(_window, true);
|
||||
ImGui_ImplOpenGL3_Init(_GLSL_version);
|
||||
|
||||
const std::filesystem::path filefont = "../../fonts/Sweet16mono.ttf";
|
||||
|
||||
if (std::filesystem::exists(filefont))
|
||||
ImGui::GetIO().Fonts->AddFontFromFileTTF(filefont.c_str(), 16);
|
||||
else
|
||||
ImGui::GetIO().Fonts->AddFontDefault();
|
||||
}
|
||||
}
|
||||
|
||||
@ -107,11 +116,12 @@ void
|
||||
MHacheWindow::build_widgets()
|
||||
{
|
||||
ImGuiWindowFlags window_flags = 0;
|
||||
window_flags |= ImGuiWindowFlags_NoBackground;
|
||||
window_flags |= ImGuiWindowFlags_NoDecoration;
|
||||
window_flags |= ImGuiWindowFlags_MenuBar;
|
||||
window_flags |= ImGuiWindowFlags_NoSavedSettings;
|
||||
window_flags |= ImGuiWindowFlags_NoMove;
|
||||
window_flags |= ImGuiWindowFlags_NoResize;
|
||||
window_flags |= ImGuiWindowFlags_NoCollapse;
|
||||
window_flags |= ImGuiWindowFlags_NoTitleBar;
|
||||
|
||||
const ImGuiViewport* main_viewport = ImGui::GetMainViewport();
|
||||
if (main_viewport) {
|
||||
@ -141,8 +151,7 @@ MHacheWindow::build_widgets()
|
||||
|
||||
ImGui::TextColored(
|
||||
ImVec4(rgba[0], rgba[1], rgba[2], rgba[3]),
|
||||
"%s",
|
||||
skill.name().c_str()
|
||||
"%s", skill.name().c_str()
|
||||
);
|
||||
|
||||
if (ImGui::IsItemHovered()) {
|
||||
@ -186,6 +195,9 @@ MHacheWindow::build_menu()
|
||||
void
|
||||
MHacheWindow::resize(int width, int height)
|
||||
{
|
||||
_width = width;
|
||||
_height = height;
|
||||
|
||||
if (!ready()) return;
|
||||
|
||||
GLFWmonitor* monitor = glfwGetPrimaryMonitor();
|
||||
@ -194,20 +206,18 @@ MHacheWindow::resize(int width, int height)
|
||||
const GLFWvidmode* videomode = glfwGetVideoMode(monitor);
|
||||
if (!videomode) return;
|
||||
|
||||
_width = width;
|
||||
_height = height;
|
||||
|
||||
glfwSetWindowPos(_window, (videomode->width - _width) * 0.5, (videomode->height - _height) * 0.5);
|
||||
glfwSetWindowSize(_window, width, height);
|
||||
glfwSetWindowSize(_window, _width, _height);
|
||||
}
|
||||
|
||||
bool
|
||||
MHacheWindow::ready()
|
||||
{
|
||||
return (
|
||||
!(SIGINT == _signal_status || SIGTERM == _signal_status || SIGABRT == _signal_status)
|
||||
&& _window && !glfwWindowShouldClose(_window)
|
||||
);
|
||||
return SIGINT != _signal_status
|
||||
&& SIGTERM != _signal_status
|
||||
&& SIGABRT != _signal_status
|
||||
&& _window
|
||||
&& !glfwWindowShouldClose(_window);
|
||||
}
|
||||
|
||||
void
|
||||
|
Loading…
x
Reference in New Issue
Block a user