2024-04-26 15:37:28 +08:00
|
|
|
#pragma once
|
2024-05-28 19:59:41 +08:00
|
|
|
#include "YYCCInternal.hpp"
|
|
|
|
|
|
|
|
#include "EncodingHelper.hpp"
|
2024-04-26 15:37:28 +08:00
|
|
|
#include <string>
|
|
|
|
#include <cinttypes>
|
2024-05-28 19:59:41 +08:00
|
|
|
#include <type_traits>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <limits>
|
2024-04-26 15:37:28 +08:00
|
|
|
|
|
|
|
namespace YYCC::ParserHelper {
|
|
|
|
|
2024-05-28 19:59:41 +08:00
|
|
|
template<class>
|
|
|
|
constexpr bool g_AlwaysFalse = false;
|
|
|
|
|
|
|
|
template<typename _Ty, std::enable_if_t<std::is_floating_point_v<_Ty>, int> = 0>
|
|
|
|
bool TryParse(const std::string& strl, _Ty& num) {
|
|
|
|
try {
|
|
|
|
// float types
|
|
|
|
if constexpr (std::is_same_v<_Ty, float>) {
|
|
|
|
num = std::stof(strl, nullptr);
|
|
|
|
} else if constexpr (std::is_same_v<_Ty, double>) {
|
|
|
|
num = std::stod(strl, nullptr);
|
|
|
|
} else if constexpr (std::is_same_v<_Ty, long double>) {
|
|
|
|
num = std::stold(strl, nullptr);
|
|
|
|
} else {
|
|
|
|
static_assert(g_AlwaysFalse<_Ty>, "Invalid float type.");
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} catch (const std::invalid_argument&) {
|
|
|
|
return false;
|
|
|
|
} catch (const std::out_of_range&) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
template<typename _Ty, std::enable_if_t<std::is_integral_v<_Ty> && !std::is_same_v<_Ty, bool>, int> = 0>
|
|
|
|
bool TryParse(const std::string& strl, _Ty& num, int base = 10) {
|
|
|
|
try {
|
|
|
|
// integer type
|
|
|
|
// decide integer type
|
|
|
|
using container_t = std::conditional_t<std::is_unsigned_v<_Ty>, unsigned long long, long long>;
|
|
|
|
// parse it from string according to whether integer type is signed.
|
|
|
|
container_t cache;
|
|
|
|
if constexpr (std::is_unsigned_v<_Ty>) {
|
|
|
|
cache = std::stoull(strl, nullptr, base);
|
|
|
|
} else {
|
|
|
|
cache = std::stoll(strl, nullptr, base);
|
|
|
|
}
|
|
|
|
// check its range
|
|
|
|
if (cache < std::numeric_limits<_Ty>::min() || cache > std::numeric_limits<_Ty>::max())
|
|
|
|
return false;
|
|
|
|
num = static_cast<_Ty>(cache);
|
|
|
|
return true;
|
|
|
|
} catch (const std::invalid_argument&) {
|
|
|
|
return false;
|
|
|
|
} catch (const std::out_of_range&) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2024-05-30 14:53:46 +08:00
|
|
|
template<typename _Ty, std::enable_if_t<std::is_same_v<_Ty, bool>, int> = 0>
|
|
|
|
bool TryParse(const std::string& strl, _Ty& num) {
|
2024-05-28 19:59:41 +08:00
|
|
|
if (strl == "true") num = true;
|
|
|
|
else if (strl == "false") num = false;
|
|
|
|
else return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename _Ty, std::enable_if_t<std::is_arithmetic_v<_Ty>, int> = 0>
|
|
|
|
_Ty Parse(const std::string& strl) {
|
|
|
|
_Ty ret;
|
|
|
|
TryParse(strl, ret);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename _Ty, std::enable_if_t<std::is_arithmetic_v<_Ty>, int> = 0>
|
|
|
|
std::string ToString(_Ty num) {
|
|
|
|
return std::to_string(num);
|
|
|
|
}
|
|
|
|
template<>
|
|
|
|
std::string ToString<bool>(bool num) {
|
|
|
|
if (num) return std::string("true");
|
|
|
|
else return std::string("false");
|
|
|
|
}
|
2024-04-26 15:37:28 +08:00
|
|
|
|
|
|
|
}
|