|
|
@ -77,7 +77,7 @@ class ini_config
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<std::integral int_type>
|
|
|
|
template<std::integral int_type>
|
|
|
|
consteval static int_type from_string(const char_type *str) noexcept {
|
|
|
|
constexpr static int_type from_string(const char_type *str) noexcept {
|
|
|
|
int_type ret = 0;
|
|
|
|
int_type ret = 0;
|
|
|
|
bool neg = *str == '-';
|
|
|
|
bool neg = *str == '-';
|
|
|
|
if (neg)
|
|
|
|
if (neg)
|
|
|
@ -87,7 +87,7 @@ class ini_config
|
|
|
|
return !neg ? ret : -ret;
|
|
|
|
return !neg ? ret : -ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
template<std::floating_point float_type>
|
|
|
|
template<std::floating_point float_type>
|
|
|
|
consteval static float_type from_string(const char_type *str) noexcept {
|
|
|
|
constexpr static float_type from_string(const char_type *str) noexcept {
|
|
|
|
float_type ret = 0;
|
|
|
|
float_type ret = 0;
|
|
|
|
bool neg = *str == '-';
|
|
|
|
bool neg = *str == '-';
|
|
|
|
if (neg)
|
|
|
|
if (neg)
|
|
|
@ -357,10 +357,7 @@ public:
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
constexpr auto end(const char_type *section) const noexcept {
|
|
|
|
constexpr auto end(const char_type *section) const noexcept {
|
|
|
|
auto it = begin(section);
|
|
|
|
auto it = begin(section);
|
|
|
|
while (++it != end()) {
|
|
|
|
while (++it != end() && stringmatch(it->section, section));
|
|
|
|
if (!stringmatch(it->section, section))
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return it;
|
|
|
|
return it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -386,8 +383,8 @@ public:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Finds and returns the value paired with the given key.
|
|
|
|
* Returns the value for the given key as a string.
|
|
|
|
* Returns an empty string on failure.
|
|
|
|
* Returns "" if key does not exist.
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
consteval auto get(const char_type *key) const noexcept {
|
|
|
|
consteval auto get(const char_type *key) const noexcept {
|
|
|
|
for (auto kvp : *this) {
|
|
|
|
for (auto kvp : *this) {
|
|
|
@ -397,17 +394,16 @@ public:
|
|
|
|
return "";
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Finds and returns the value paired with the given key,
|
|
|
|
* Returns the value for the given key, converted to the given type.
|
|
|
|
* converting it to the specified integral or floating-point type.
|
|
|
|
* Returns zero if key does not exist.
|
|
|
|
* Returns zero on failure.
|
|
|
|
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
template<typename T> requires(std::integral<T> || std::floating_point<T>)
|
|
|
|
template<typename T> requires(std::integral<T> || std::floating_point<T>)
|
|
|
|
consteval T get(const char_type *key) const noexcept {
|
|
|
|
consteval T get(const char_type *key) const noexcept {
|
|
|
|
return from_string<T>(get(key));
|
|
|
|
return from_string<T>(get(key));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Finds and returns the value paired with the given key, in the given section.
|
|
|
|
* Returns the value for the given key in the given section.
|
|
|
|
* Returns an empty string on failure.
|
|
|
|
* Returns "" on failure.
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
consteval auto get(const char_type *sec, const char_type *key) const noexcept {
|
|
|
|
consteval auto get(const char_type *sec, const char_type *key) const noexcept {
|
|
|
|
for (auto kvp : section(sec)) {
|
|
|
|
for (auto kvp : section(sec)) {
|
|
|
@ -417,8 +413,8 @@ public:
|
|
|
|
return "";
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Finds and returns the value paired with the given key in the given section,
|
|
|
|
* Returns the value for the given key in the given section,
|
|
|
|
* converting it to the specified integral or floating-point type.
|
|
|
|
* converting it to the specified type.
|
|
|
|
* Returns zero on failure.
|
|
|
|
* Returns zero on failure.
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
template<typename T> requires(std::integral<T> || std::floating_point<T>)
|
|
|
|
template<typename T> requires(std::integral<T> || std::floating_point<T>)
|
|
|
@ -426,12 +422,53 @@ public:
|
|
|
|
return from_string<T>(get(sec, key));
|
|
|
|
return from_string<T>(get(sec, key));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Attempts to get the value for the given key.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
auto tryget(const char_type *key) const noexcept {
|
|
|
|
|
|
|
|
for (auto kvp : *this) {
|
|
|
|
|
|
|
|
if (stringmatch(kvp.first, key))
|
|
|
|
|
|
|
|
return kvp.second;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Attempts to get the value for the given key, converted to the given type.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
template<typename T> requires(std::integral<T> || std::floating_point<T>)
|
|
|
|
|
|
|
|
T tryget(const char_type *key) const noexcept {
|
|
|
|
|
|
|
|
return from_string<T>(tryget(key));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Attempts to get the value for the given key.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
auto tryget(const char_type *sec, const char_type *key) const noexcept {
|
|
|
|
|
|
|
|
for (auto kvp : section(sec)) {
|
|
|
|
|
|
|
|
if (stringmatch(kvp.first, key))
|
|
|
|
|
|
|
|
return kvp.second;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
|
|
* Attempts to get the value for the given key, converted to the given type.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
template<typename T> requires(std::integral<T> || std::floating_point<T>)
|
|
|
|
|
|
|
|
T tryget(const char_type *sec, const char_type *key) const noexcept {
|
|
|
|
|
|
|
|
return from_string<T>(tryget(sec, key));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
consteval bool contains(const char_type *key) const noexcept {
|
|
|
|
consteval bool contains(const char_type *key) const noexcept {
|
|
|
|
return *get(key) != '\0';
|
|
|
|
return *get(key) != '\0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
consteval bool contains(const char_type *sec, const char_type *key) const noexcept {
|
|
|
|
consteval bool contains(const char_type *sec, const char_type *key) const noexcept {
|
|
|
|
return *get(sec, key) != '\0';
|
|
|
|
return *get(sec, key) != '\0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool trycontains(const char_type *key) const noexcept {
|
|
|
|
|
|
|
|
return *tryget(key) != '\0';
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool trycontains(const char_type *sec, const char_type *key) const noexcept {
|
|
|
|
|
|
|
|
return *tryget(sec, key) != '\0';
|
|
|
|
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace ini_config
|
|
|
|
} // namespace ini_config
|
|
|
@ -445,4 +482,9 @@ consteval auto operator ""_ini()
|
|
|
|
return ini_config::ini_config<Input>();
|
|
|
|
return ini_config::ini_config<Input>();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// For MSVC, the below alternative seems promising, though
|
|
|
|
|
|
|
|
// MSVC v19.28 complains about running out of heap.
|
|
|
|
|
|
|
|
//template <ini_config::string_container Input>
|
|
|
|
|
|
|
|
//constexpr auto make_ini_config = ini_config::ini_config<Input>();
|
|
|
|
|
|
|
|
|
|
|
|
#endif // TCSULLIVAN_INI_CONFIG_HPP
|
|
|
|
#endif // TCSULLIVAN_INI_CONFIG_HPP
|
|
|
|