#pragma once #include "StarFormat.hpp" #include "StarString.hpp" #include "StarStringView.hpp" #include "StarMaybe.hpp" #include "fast_float.h" namespace Star { STAR_EXCEPTION(BadLexicalCast, StarException); void throwLexicalCastError(std::errc ec, const char* first, const char* last); template bool tryLexicalCast(Type& result, const char* first, const char* last) { auto res = fast_float::from_chars(first, last, result); return res.ptr == last && (res.ec == std::errc() || res.ec == std::errc::result_out_of_range); } template <> bool tryLexicalCast(bool& result, const char* first, const char* last); template bool tryLexicalCast(Type& result, String const& s) { return tryLexicalCast(result, s.utf8Ptr(), s.utf8Ptr() + s.utf8Size()); } template bool tryLexicalCast(Type& result, StringView s) { return tryLexicalCast(result, s.utf8Ptr(), s.utf8Ptr() + s.utf8Size()); } template Maybe maybeLexicalCast(const char* first, const char* last) { Type result{}; if (tryLexicalCast(result, first, last)) return result; else return {}; } template Maybe maybeLexicalCast(StringView s) { return maybeLexicalCast(s.utf8Ptr(), s.utf8Ptr() + s.utf8Size()); } template Type lexicalCast(const char* first, const char* last) { Type result{}; auto res = fast_float::from_chars(first, last, result); if ((res.ec != std::errc() && res.ec != std::errc::result_out_of_range) || res.ptr != last) throwLexicalCastError(res.ec, first, last); return result; } template <> bool lexicalCast(const char* first, const char* last); template Type lexicalCast(StringView s) { return lexicalCast(s.utf8Ptr(), s.utf8Ptr() + s.utf8Size()); } }