2024-07-02 10:52:24 +08:00
|
|
|
#pragma once
|
|
|
|
#include "YYCCInternal.hpp"
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
|
|
|
#include <initializer_list>
|
|
|
|
#include <type_traits>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <functional>
|
|
|
|
#include <stdexcept>
|
2024-07-22 22:41:10 +08:00
|
|
|
#include <cstring>
|
2024-07-02 10:52:24 +08:00
|
|
|
|
|
|
|
namespace YYCC::ConfigManager {
|
|
|
|
|
|
|
|
template<typename _Ty>
|
2024-07-05 22:25:14 +08:00
|
|
|
struct Constraint {
|
2024-07-02 10:52:24 +08:00
|
|
|
using CheckFct_t = std::function<bool(const _Ty&)>;
|
|
|
|
//using CorrectFct_t = std::function<_Ty(const _Ty&)>;
|
|
|
|
CheckFct_t m_CheckFct;
|
|
|
|
//CorrectFct_t m_CorrectFct;
|
|
|
|
|
|
|
|
bool IsValid() const {
|
|
|
|
return m_CheckFct != nullptr/* && m_CorrectFct != nullptr*/;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-07-05 22:25:14 +08:00
|
|
|
namespace ConstraintPresets {
|
2024-07-02 10:52:24 +08:00
|
|
|
|
2024-07-03 10:14:17 +08:00
|
|
|
template<typename _Ty, std::enable_if_t<std::is_arithmetic_v<_Ty> && !std::is_enum_v<_Ty> && !std::is_same_v<_Ty, bool>, int> = 0>
|
2024-07-05 22:25:14 +08:00
|
|
|
Constraint<_Ty> GetNumberRangeConstraint(_Ty min_value, _Ty max_value) {
|
2024-07-02 10:52:24 +08:00
|
|
|
if (min_value > max_value)
|
2024-07-05 22:25:14 +08:00
|
|
|
throw std::invalid_argument("invalid min max value for NumberRangeConstraint");
|
|
|
|
return Constraint<_Ty> {
|
2024-07-02 10:52:24 +08:00
|
|
|
[min_value, max_value](const _Ty& val) -> bool { return (val <= max_value) && (val >= min_value); }
|
|
|
|
/*[min_value, max_value](const _Ty& val) -> _Ty { return std::clamp(val, min_value, max_value); }*/
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
class AbstractSetting {
|
|
|
|
friend class CoreManager;
|
|
|
|
public:
|
|
|
|
AbstractSetting(const yycc_char8_t* name) : m_Name(), m_RawData() {
|
|
|
|
if (name != nullptr) m_Name = name;
|
|
|
|
}
|
|
|
|
virtual ~AbstractSetting() {}
|
|
|
|
|
|
|
|
// Name interface
|
|
|
|
public:
|
|
|
|
const yycc_u8string& GetName() const { return m_Name; }
|
|
|
|
private:
|
|
|
|
yycc_u8string m_Name;
|
|
|
|
|
|
|
|
// User Implementations
|
|
|
|
protected:
|
|
|
|
virtual bool UserLoad() = 0;
|
|
|
|
virtual bool UserSave() = 0;
|
|
|
|
virtual void UserReset() = 0;
|
|
|
|
|
|
|
|
// Buffer related functions
|
|
|
|
protected:
|
|
|
|
void ResizeData(size_t new_size) { m_RawData.resize(new_size); }
|
|
|
|
const void* GetDataPtr() const { return m_RawData.data(); }
|
|
|
|
void* GetDataPtr() { return m_RawData.data(); }
|
|
|
|
size_t GetDataSize() const { return m_RawData.size(); }
|
|
|
|
private:
|
|
|
|
std::vector<uint8_t> m_RawData;
|
|
|
|
};
|
|
|
|
|
|
|
|
class CoreManager {
|
|
|
|
public:
|
|
|
|
CoreManager(
|
|
|
|
const yycc_char8_t* cfg_file_path,
|
|
|
|
uint64_t version_identifier,
|
|
|
|
std::initializer_list<AbstractSetting*> settings);
|
|
|
|
~CoreManager() {}
|
|
|
|
|
|
|
|
// Core functions
|
|
|
|
public:
|
|
|
|
bool Load();
|
|
|
|
bool Save();
|
|
|
|
void Reset();
|
|
|
|
|
|
|
|
private:
|
|
|
|
using FileHandleGuard_t = std::unique_ptr<FILE, std::function<void(FILE*)>>;
|
|
|
|
FileHandleGuard_t GetFileHandle(const yycc_char8_t* mode) const;
|
|
|
|
|
|
|
|
yycc_u8string m_CfgFilePath;
|
|
|
|
uint64_t m_VersionIdentifier;
|
|
|
|
std::map<yycc_u8string, AbstractSetting*> m_Settings;
|
|
|
|
};
|
|
|
|
|
|
|
|
#pragma region Setting Presets
|
|
|
|
|
2024-07-03 10:14:17 +08:00
|
|
|
template<typename _Ty, std::enable_if_t<std::is_arithmetic_v<_Ty> || std::is_enum_v<_Ty>, int> = 0>
|
2024-07-02 10:52:24 +08:00
|
|
|
class NumberSetting : public AbstractSetting {
|
|
|
|
public:
|
2024-07-05 22:25:14 +08:00
|
|
|
NumberSetting(const yycc_char8_t* name, _Ty default_value, Constraint<_Ty> constraint = Constraint<_Ty> {}) :
|
|
|
|
AbstractSetting(name), m_Data(default_value), m_DefaultData(default_value), m_Constraint(constraint) {}
|
2024-07-02 10:52:24 +08:00
|
|
|
virtual ~NumberSetting() {}
|
|
|
|
|
|
|
|
_Ty Get() const { return m_Data; }
|
2024-07-07 21:09:51 +08:00
|
|
|
bool Set(_Ty new_data) {
|
2024-07-02 10:52:24 +08:00
|
|
|
// validate data
|
2024-07-05 22:25:14 +08:00
|
|
|
if (m_Constraint.IsValid() && !m_Constraint.m_CheckFct(new_data))
|
2024-07-02 10:52:24 +08:00
|
|
|
return false;
|
|
|
|
// assign data
|
|
|
|
m_Data = new_data;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual bool UserLoad() override {
|
|
|
|
// read data
|
|
|
|
if (sizeof(m_Data) != GetDataSize())
|
|
|
|
return false;
|
|
|
|
m_Data = *reinterpret_cast<const _Ty*>(GetDataPtr());
|
|
|
|
// check data
|
2024-07-05 22:25:14 +08:00
|
|
|
if (m_Constraint.IsValid() && !m_Constraint.m_CheckFct(m_Data))
|
2024-07-02 10:52:24 +08:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
virtual bool UserSave() override {
|
|
|
|
// write data
|
|
|
|
ResizeData(sizeof(m_Data));
|
|
|
|
*reinterpret_cast<_Ty*>(GetDataPtr()) = m_Data;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
virtual void UserReset() override {
|
|
|
|
m_Data = m_DefaultData;
|
|
|
|
}
|
|
|
|
|
|
|
|
_Ty m_Data, m_DefaultData;
|
2024-07-05 22:25:14 +08:00
|
|
|
Constraint<_Ty> m_Constraint;
|
2024-07-02 10:52:24 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
class StringSetting : public AbstractSetting {
|
|
|
|
public:
|
2024-07-07 21:09:51 +08:00
|
|
|
StringSetting(const yycc_char8_t* name, const yycc_u8string_view& default_value, Constraint<yycc_u8string_view> constraint = Constraint<yycc_u8string_view> {}) :
|
2024-07-05 22:25:14 +08:00
|
|
|
AbstractSetting(name), m_Data(), m_DefaultData(), m_Constraint(constraint) {
|
2024-07-07 21:09:51 +08:00
|
|
|
m_Data = default_value;
|
|
|
|
m_DefaultData = default_value;
|
2024-07-02 10:52:24 +08:00
|
|
|
}
|
|
|
|
virtual ~StringSetting() {}
|
|
|
|
|
|
|
|
const yycc_u8string& Get() const { return m_Data; }
|
2024-07-07 21:09:51 +08:00
|
|
|
bool Set(const yycc_u8string_view& new_data) {
|
2024-07-02 10:52:24 +08:00
|
|
|
// check data validation
|
2024-07-07 21:09:51 +08:00
|
|
|
if (m_Constraint.IsValid() && !m_Constraint.m_CheckFct(new_data))
|
2024-07-02 10:52:24 +08:00
|
|
|
return false;
|
|
|
|
// assign data
|
|
|
|
m_Data = new_data;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual bool UserLoad() override {
|
|
|
|
// read string length
|
|
|
|
size_t string_length;
|
|
|
|
if (GetDataSize() < sizeof(string_length))
|
|
|
|
return false;
|
|
|
|
string_length = *reinterpret_cast<const size_t*>(GetDataPtr());
|
|
|
|
// read string body
|
|
|
|
if (GetDataSize() != sizeof(string_length) + string_length)
|
|
|
|
return false;
|
|
|
|
m_Data.assign(
|
|
|
|
reinterpret_cast<const yycc_char8_t*>(static_cast<const uint8_t*>(GetDataPtr()) + sizeof(string_length)),
|
|
|
|
string_length
|
|
|
|
);
|
|
|
|
// check data
|
2024-07-05 22:25:14 +08:00
|
|
|
if (m_Constraint.IsValid() && !m_Constraint.m_CheckFct(m_Data))
|
2024-07-02 10:52:24 +08:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
virtual bool UserSave() override {
|
|
|
|
// allocate result buffer
|
|
|
|
size_t string_length = m_Data.size();
|
|
|
|
ResizeData(sizeof(string_length) + string_length);
|
|
|
|
// get pointer
|
|
|
|
uint8_t* ptr = static_cast<uint8_t*>(GetDataPtr());
|
|
|
|
// assign string length
|
|
|
|
*reinterpret_cast<size_t*>(ptr) = string_length;
|
|
|
|
// assign string body
|
|
|
|
std::memcpy(ptr + sizeof(string_length), m_Data.data(), string_length);
|
|
|
|
return true;
|
|
|
|
}
|
2024-07-03 10:14:17 +08:00
|
|
|
virtual void UserReset() override {
|
|
|
|
m_Data = m_DefaultData;
|
|
|
|
}
|
2024-07-02 10:52:24 +08:00
|
|
|
|
|
|
|
yycc_u8string m_Data, m_DefaultData;
|
2024-07-07 21:09:51 +08:00
|
|
|
Constraint<yycc_u8string_view> m_Constraint;
|
2024-07-02 10:52:24 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
#pragma endregion
|
|
|
|
|
|
|
|
|
|
|
|
}
|