#pragma once #include "../concepts.hpp" #include #include #include #define GZ_UTIL_STRING_CONCEPTS /** * @brief Declaration of toString in global namespace, so that concepts can use it. */ template std::string toString(const T& s); namespace gz::util { /** * @name 1) concepts for elementary types and custom toStrings() * @details * Definition order matters, the earlier the concept, the "stronger" because later concepts must not also satisfy the previous ones. * @{ */ /// same as std::string, std::string_view or const char* template concept Stringy = util::IsInPack; /// can construct std::string from T template concept CanConstructString = !Stringy && requires(const T& t) { { std::string(t) } -> std::same_as; }; /// has .toString() const member template concept HasToStringMember = !Stringy && !CanConstructString && requires(const T& t) { { t.toString() }-> Stringy; }; /// has .to_string() const member template concept HasToStringMember2 = !Stringy && !CanConstructString && !HasToStringMember && requires(const T& t) { { t.to_string() }-> Stringy; }; /// works with std::to_string() template concept WorksWithStdToString = !Stringy && !CanConstructString && !HasToStringMember && !HasToStringMember2 && requires(const T& t) { { std::to_string(t) } -> Stringy; }; /// toString function overload exists in global namespace and returns std::string template concept ConvertibleToStringGlobal = !Stringy && !CanConstructString && !HasToStringMember && !HasToStringMember2 && !WorksWithStdToString && requires(const T& t) { { ::toString(t) } -> std::same_as; }; template concept _ToStringBasicNoPtr = Stringy || CanConstructString || HasToStringMember || HasToStringMember2 || WorksWithStdToString || ConvertibleToStringGlobal; #ifndef GZ_TO_STRING_NO_VECTORS // VECTOR /// Type having string-convertible x, y members and sizeof(T) == 2 * sizeof(x) template concept Vector2ConvertibleToString = !_ToStringBasicNoPtr && requires(T t) { { t.x } -> _ToStringBasicNoPtr; { t.y } -> _ToStringBasicNoPtr; requires sizeof(t.x) * 2 == sizeof(T); }; /// Type having string-convertible x, y, z members and sizeof(T) == 3 * sizeof(x) template concept Vector3ConvertibleToString = !_ToStringBasicNoPtr && requires(T t) { { t.x } -> _ToStringBasicNoPtr; { t.y } -> _ToStringBasicNoPtr; { t.z } -> _ToStringBasicNoPtr; requires sizeof(t.x) * 3 == sizeof(T); }; /// Type having string-convertible x, y, z, w members and sizeof(T) == 4 * sizeof(x) template concept Vector4ConvertibleToString = !_ToStringBasicNoPtr && requires(T t) { { t.x } -> _ToStringBasicNoPtr; { t.y } -> _ToStringBasicNoPtr; { t.z } -> _ToStringBasicNoPtr; { t.w } -> _ToStringBasicNoPtr; requires sizeof(t.x) * 4 == sizeof(T); }; /// Type having string-convertible width, height members and sizeof(T) == 2 * sizeof(width) template concept Extent2DConvertibleToString = !_ToStringBasicNoPtr && requires(T t) { { t.width } -> _ToStringBasicNoPtr; { t.height } -> _ToStringBasicNoPtr; requires sizeof(t.width) * 2 == sizeof(T); }; /// Type having string-convertible width, height, depth members and sizeof(T) == 3 * sizeof(width) template concept Extent3DConvertibleToString = !_ToStringBasicNoPtr && requires(T t) { { t.width } -> _ToStringBasicNoPtr; { t.height } -> _ToStringBasicNoPtr; { t.depth } -> _ToStringBasicNoPtr; requires sizeof(t.width) * 3 == sizeof(T); }; #endif template concept _ToStringBasicOrVectorNoPtr = _ToStringBasicNoPtr #ifndef GZ_TO_STRING_NO_VECTORS || Vector2ConvertibleToString || Vector3ConvertibleToString || Vector4ConvertibleToString || Extent2DConvertibleToString || Extent3DConvertibleToString #endif ; template concept PtrToToStringBasicOrVector = !_ToStringBasicOrVectorNoPtr && _ToStringBasicOrVectorNoPtr>; /// @} } // namespace gz::util namespace gz { /** * @name 1) Converting a type to string * @details * All toString functions for types that satisfy util::ToStringBasicNoPtr or util::PtrToToStringBasicType * @{ */ /** * @brief Return the string * @returns static_cast(t) */ template inline std::string toString(const T& t) { return static_cast(t); } /** * @brief Construct a string from a string like-type * @returns std::string(t) */ template inline std::string toString(const T& t) { return std::string(t); } /** * @overload * @brief Construct a string from a type having a toString() const member function * @returns t.toString() */ template inline std::string toString(const T& t) { return t.toString(); } /** * @overload * @brief Construct a string from a type having a to_string() const member function * @returns t.to_string() */ template inline std::string toString(const T& t) { return t.to_string(); } /** * @overload * @brief Construct a string from a number * @returns std::to_string(t) */ template inline std::string toString(const T& t) { return std::to_string(t); } /** * @brief Construct a string from a boolean * @details * Unlike std::to_string(bool), which returns 0 or 1, this function returns "true" or "false" * @returns "true" or "false" */ template T> inline std::string toString(const T& b) { return b ? "true" : "false"; } /** * @overload * @brief Construct a string from a type that has toString declared in global namespace */ template inline std::string toString(const T& t) { return ::toString(t); } #ifndef GZ_TO_STRING_NO_VECTORS /** * @overload * @brief Construct a string from a vector with x, y members * @returns ( x, y ) */ template inline std::string toString(const T& t) { std::string s = "( "; s += toString(t.x) + ", "; s += toString(t.y) + " )"; return s; } /** * @overload * @brief Construct a string from a vector with x, y members * @returns ( x, y, z ) */ template inline std::string toString(const T& t) { std::string s = "( "; s += toString(t.x) + ", "; s += toString(t.y) + ", "; s += toString(t.z) + " )"; return s; } /** * @overload * @brief Construct a string from a vector with x, y, z, w members * @returns ( x, y, z, w ) */ template inline std::string toString(const T& t) { std::string s = "( "; s += toString(t.x) + ", "; s += toString(t.y) + ", "; s += toString(t.z) + ", "; s += toString(t.w) + " )"; return s; } /** * @overload * @brief Construct a string from a type having width and height members * @returns ( width, height ) */ template inline std::string toString(const T& t) { std::string s = "( "; s += toString(t.width) + ", "; s += toString(t.height) + " ) "; return s; } /** * @overload * @brief Construct a string from a type having width and height members * @returns ( width, height, depth ) */ template inline std::string toString(const T& t) { std::string s = "( "; s += toString(t.width) + ", "; s += toString(t.height) + ", "; s += toString(t.depth) + " ) "; return s; } #endif /** * @overload * @brief Construct a string from the element the pointer points at * @returns *t */ template inline std::string toString(const T& t) { return toString(*t); } } // namespace gz namespace gz::util { /** * @name 2) concepts for containers that hold toString() compatible types * @{ */ /// Everything that is convertible to string with toString at this point template concept _ConvertibleToString = requires (const T& t) { { gz::toString(t) } -> std::same_as; }; // CONTAINER /// Forward range having string-convertible elements template concept ForwardRangeConvertibleToString = !_ConvertibleToString and std::ranges::forward_range and _ConvertibleToString>; template concept _TypeOrForwardRangeConvertibleToString = _ConvertibleToString || ForwardRangeConvertibleToString; /// Pair having string-convertible elements template concept PairConvertibleToString = !ForwardRangeConvertibleToString and requires(const T& p) { requires _TypeOrForwardRangeConvertibleToString; requires _TypeOrForwardRangeConvertibleToString; }; /// Container having string-convertible pairs template concept MapConvertibleToString = !PairConvertibleToString and !_TypeOrForwardRangeConvertibleToString and std::ranges::forward_range and PairConvertibleToString>; template concept _ContainerTypeConvertibleToString = PairConvertibleToString || ForwardRangeConvertibleToString || MapConvertibleToString; /// @} } // namespace gz::util namespace gz { /** * @name 2) Converting a container to string * @details * All toString functions for types that satisfy util::_ContainerTypeConvertibleToString * @{ */ /** * @overload * @brief Construct a string from a forward range * @returns [ x1, x2, ... ] */ template std::string toString(const T& t) { std::string s = "[ "; for (auto it = t.begin(); it != t.end(); it++) { s += toString(*it) + ", "; } if (s.size() > 2) { s.erase(s.size() - 2); } s += " ]"; return s; } /** * @overload * @brief Construct a string from a pair * @returns ( first, second ) */ template inline std::string toString(const T& t) { return "( " + toString(t.first) + ", " + toString(t.second) + " )"; } /** * @overload * @brief Construct a string from a forward range holding a pair, eg a map * @returns { first: second, first: second, ... } */ template std::string toString(const T& t) { std::string s = "{ "; for (const auto& [k, v] : t) { s += toString(k) + ": "; s += toString(v) + ", "; } if (s.size() > 2) { s.erase(s.size() - 2); } s += " }"; return s; } /// @} /** * @brief Any type where gz::toString(t) exists */ template concept ConvertibleToString = requires (const T& t) { { gz::toString(t) } -> std::same_as; }; } // namespace gz /** * @file * @brief Contains functions to convert types to string */