/*============================================================================= Copyright (c) 2011-2013 Bolero MURAKAMI https://github.com/bolero-MURAKAMI/Sprout Distributed under the Boost Software License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) =============================================================================*/ #ifndef SPROUT_STRING_FLOAT_TO_STRING_HPP #define SPROUT_STRING_FLOAT_TO_STRING_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include HDR_ALGORITHM_MIN_MAX_SSCRISK_CEL_OR_SPROUT namespace sprout { namespace detail { SPROUT_STATIC_CONSTEXPR std::size_t decimal_places_length = 6; } // namespace detail // // printed_float_digits // template struct printed_float_digits : public std::integral_constant< std::size_t, sprout::numeric_limits::max_exponent10 + sprout::detail::decimal_places_length + 2 > {}; namespace detail { template inline SPROUT_CONSTEXPR sprout::basic_string::value> float_to_string_impl(FloatType val, bool negative, int digits, int v, sprout::index_tuple) { typedef sprout::detail::string_construct_access::value> access_type; return negative ? access_type::raw_construct( static_cast(digits + 2 + sprout::detail::decimal_places_length), static_cast('-'), (Indexes < digits ? sprout::detail::int_to_char(sprout::detail::float_digit_at(val, digits - 1 - Indexes)) : Indexes == digits ? static_cast('.') : Indexes < digits + 1 + sprout::detail::decimal_places_length ? sprout::detail::int_to_char(sprout::detail::int_digit_at(v, digits + sprout::detail::decimal_places_length - Indexes)) : Elem() )... ) : access_type::raw_construct( static_cast(digits + 1 + sprout::detail::decimal_places_length), (Indexes < digits ? sprout::detail::int_to_char(sprout::detail::float_digit_at(val, digits - 1 - Indexes)) : Indexes == digits ? static_cast('.') : Indexes < digits + 1 + sprout::detail::decimal_places_length ? sprout::detail::int_to_char(sprout::detail::int_digit_at(v, digits + sprout::detail::decimal_places_length - Indexes)) : Elem() )... ) ; } template inline SPROUT_CONSTEXPR sprout::basic_string::value> float_to_string(FloatType val, bool negative, int digits) { return sprout::detail::float_to_string_impl( val, negative, digits, static_cast((val - sprout::math::floor(val)) * sprout::detail::int_pow(sprout::detail::decimal_places_length)), sprout::make_index_tuple::value - 1>::make() ); } } // namespace detail // // float_to_string // template< typename Elem, typename FloatType, typename sprout::enabler_if::value>::type = sprout::enabler > inline SPROUT_CONSTEXPR sprout::basic_string::value> float_to_string(FloatType val) { typedef sprout::detail::string_construct_access::value> access_type; return sprout::math::isinf(val) ? sprout::math::signbit(val) ? access_type::raw_construct(4, static_cast('-'), static_cast('i'), static_cast('n'), static_cast('f')) : access_type::raw_construct(3, static_cast('i'), static_cast('n'), static_cast('f')) : sprout::math::isnan(val) ? sprout::math::signbit(val) ? access_type::raw_construct(4, static_cast('-'), static_cast('n'), static_cast('a'), static_cast('n')) : access_type::raw_construct(3, static_cast('n'), static_cast('a'), static_cast('n')) : sprout::detail::float_to_string( sprout::detail::float_round_at(val < 0 ? -val : val, sprout::detail::decimal_places_length), sprout::math::signbit(val), sprout::detail::float_digits_checked(val) ) ; } namespace detail { template struct printed_float_exponent10_digits : public std::integral_constant< std::size_t, NS_SSCRISK_CEL_OR_SPROUT::max(sprout::detail::int_digits(sprout::numeric_limits::max_exponent10), 2) > {}; } // namespace detail // // printed_float_exp_digits // template struct printed_float_exp_digits : public std::integral_constant< std::size_t, 5 + sprout::detail::decimal_places_length + sprout::detail::printed_float_exponent10_digits::value > {}; namespace detail { template inline SPROUT_CONSTEXPR sprout::basic_string::value> float_to_string_exp(FloatType val, bool negative, int exponent10, int e10_digits, sprout::index_tuple) { typedef sprout::detail::string_construct_access::value> access_type; return negative ? access_type::raw_construct( static_cast(5 + sprout::detail::decimal_places_length + e10_digits), static_cast('-'), (Indexes == 0 ? sprout::detail::int_to_char(sprout::detail::float_digit_at(val, 0)) : Indexes == 1 ? static_cast('.') : Indexes < 2 + sprout::detail::decimal_places_length ? sprout::detail::int_to_char(sprout::detail::float_digit_at(val, 1 - Indexes)) : Indexes == 2 + sprout::detail::decimal_places_length ? static_cast('e') : Indexes == 3 + sprout::detail::decimal_places_length ? static_cast(exponent10 < 0 ? '-' : '+') : Indexes < 4 + sprout::detail::decimal_places_length + e10_digits ? sprout::detail::int_to_char(sprout::detail::int_digit_at(exponent10, 3 + sprout::detail::decimal_places_length + e10_digits - Indexes)) : Elem() )... ) : access_type::raw_construct( static_cast(4 + sprout::detail::decimal_places_length + e10_digits), (Indexes == 0 ? sprout::detail::int_to_char(sprout::detail::float_digit_at(val, 0)) : Indexes == 1 ? static_cast('.') : Indexes < 2 + sprout::detail::decimal_places_length ? sprout::detail::int_to_char(sprout::detail::float_digit_at(val, 1 - Indexes)) : Indexes == 2 + sprout::detail::decimal_places_length ? static_cast('e') : Indexes == 3 + sprout::detail::decimal_places_length ? static_cast(exponent10 < 0 ? '-' : '+') : Indexes < 4 + sprout::detail::decimal_places_length + e10_digits ? sprout::detail::int_to_char(sprout::detail::int_digit_at(exponent10, 3 + sprout::detail::decimal_places_length + e10_digits - Indexes)) : Elem() )... ) ; } } // namespace detail // // float_to_string_exp // template< typename Elem, typename FloatType, typename sprout::enabler_if::value>::type = sprout::enabler > inline SPROUT_CONSTEXPR sprout::basic_string::value> float_to_string_exp(FloatType val) { typedef sprout::detail::string_construct_access::value> access_type; return sprout::math::isinf(val) ? sprout::math::signbit(val) ? access_type::raw_construct(4, static_cast('-'), static_cast('i'), static_cast('n'), static_cast('f')) : access_type::raw_construct(3, static_cast('i'), static_cast('n'), static_cast('f')) : sprout::math::isnan(val) ? sprout::math::signbit(val) ? access_type::raw_construct(4, static_cast('-'), static_cast('n'), static_cast('a'), static_cast('n')) : access_type::raw_construct(3, static_cast('n'), static_cast('a'), static_cast('n')) : sprout::detail::float_to_string_exp( sprout::detail::float_round_at( (val < 0 ? -val : val) / sprout::detail::float_pow10(sprout::detail::float_exponent10(val)), sprout::detail::decimal_places_length ), sprout::math::signbit(val), sprout::detail::float_exponent10(val), NS_SSCRISK_CEL_OR_SPROUT::max(sprout::detail::int_digits(sprout::detail::float_exponent10(val)), 2), sprout::make_index_tuple::value - 1>::make() ) ; } // // to_string_of // template::value>::type = sprout::enabler> inline SPROUT_CONSTEXPR sprout::basic_string::value> to_string_of(FloatType val) { return sprout::float_to_string(val); } // // to_string // template::value>::type = sprout::enabler> inline SPROUT_CONSTEXPR sprout::basic_string::value> to_string(FloatType val) { return sprout::to_string_of(val); } // // to_wstring // template::value>::type = sprout::enabler> inline SPROUT_CONSTEXPR sprout::basic_string::value> to_wstring(FloatType val) { return sprout::to_string_of(val); } // // to_u16string // template::value>::type = sprout::enabler> inline SPROUT_CONSTEXPR sprout::basic_string::value> to_u16string(FloatType val) { return sprout::to_string_of(val); } // // to_u32string // template::value>::type = sprout::enabler> inline SPROUT_CONSTEXPR sprout::basic_string::value> to_u32string(FloatType val) { return sprout::to_string_of(val); } } // namespace sprout #endif // #ifndef SPROUT_STRING_FLOAT_TO_STRING_HPP