diff options
Diffstat (limited to 'boost/convert/base.hpp')
-rw-r--r-- | boost/convert/base.hpp | 180 |
1 files changed, 180 insertions, 0 deletions
diff --git a/boost/convert/base.hpp b/boost/convert/base.hpp new file mode 100644 index 0000000000..128c46f3a5 --- /dev/null +++ b/boost/convert/base.hpp @@ -0,0 +1,180 @@ +// Copyright (c) 2009-2014 Vladimir Batov. +// Use, modification and distribution are subject to the Boost Software License, +// Version 1.0. See http://www.boost.org/LICENSE_1_0.txt. + +#ifndef BOOST_CONVERT_CONVERTER_BASE_HPP +#define BOOST_CONVERT_CONVERTER_BASE_HPP + +#include <boost/convert/parameters.hpp> +#include <boost/convert/detail/is_string.hpp> +#include <cctype> +#include <cstring> + +namespace boost { namespace cnv +{ + namespace ARG = boost::cnv::parameter; + + template<typename> struct cnvbase; +}} + +#define BOOST_CNV_TO_STRING \ + template<typename string_type> \ + typename boost::enable_if<cnv::is_string<string_type>, void>::type \ + operator() + +#define BOOST_CNV_STRING_TO \ + template<typename string_type> \ + typename boost::enable_if<cnv::is_string<string_type>, void>::type \ + operator() + +#define BOOST_CNV_PARAM(param_name, param_type) \ + derived_type& operator()(boost::parameter::aux::tag<ARG::type::param_name, param_type>::type const& arg) + +template<typename derived_type> +struct boost::cnv::cnvbase +{ + typedef cnvbase this_type; + typedef int int_type; + typedef unsigned int uint_type; + typedef long int lint_type; + typedef unsigned long int ulint_type; + typedef short int sint_type; + typedef unsigned short int usint_type; + typedef long long int llint_type; + typedef unsigned long long int ullint_type; + typedef float flt_type; + typedef double dbl_type; + typedef long double ldbl_type; + + // Integration of user-types via operator>>() + template<typename type_in, typename type_out> + void + operator()(type_in const& in, boost::optional<type_out>& out) const + { + in >> out; + } + + // Basic type to string + BOOST_CNV_TO_STRING ( int_type v, optional<string_type>& r) const { to_str_(v, r); } + BOOST_CNV_TO_STRING ( uint_type v, optional<string_type>& r) const { to_str_(v, r); } + BOOST_CNV_TO_STRING ( lint_type v, optional<string_type>& r) const { to_str_(v, r); } + BOOST_CNV_TO_STRING (ulint_type v, optional<string_type>& r) const { to_str_(v, r); } + BOOST_CNV_TO_STRING ( sint_type v, optional<string_type>& r) const { to_str_(v, r); } + BOOST_CNV_TO_STRING (usint_type v, optional<string_type>& r) const { to_str_(v, r); } + BOOST_CNV_TO_STRING ( flt_type v, optional<string_type>& r) const { to_str_(v, r); } + BOOST_CNV_TO_STRING ( dbl_type v, optional<string_type>& r) const { to_str_(v, r); } + BOOST_CNV_TO_STRING ( ldbl_type v, optional<string_type>& r) const { to_str_(v, r); } + // String to basic type + BOOST_CNV_STRING_TO (string_type const& s, optional< int_type>& r) const { str_to_(s, r); } + BOOST_CNV_STRING_TO (string_type const& s, optional< uint_type>& r) const { str_to_(s, r); } + BOOST_CNV_STRING_TO (string_type const& s, optional< lint_type>& r) const { str_to_(s, r); } + BOOST_CNV_STRING_TO (string_type const& s, optional<ulint_type>& r) const { str_to_(s, r); } + BOOST_CNV_STRING_TO (string_type const& s, optional< sint_type>& r) const { str_to_(s, r); } + BOOST_CNV_STRING_TO (string_type const& s, optional<usint_type>& r) const { str_to_(s, r); } + BOOST_CNV_STRING_TO (string_type const& s, optional< flt_type>& r) const { str_to_(s, r); } + BOOST_CNV_STRING_TO (string_type const& s, optional< dbl_type>& r) const { str_to_(s, r); } + BOOST_CNV_STRING_TO (string_type const& s, optional< ldbl_type>& r) const { str_to_(s, r); } + // Formatters +// BOOST_CNV_PARAM (locale, std::locale const) { locale_ = arg[ARG:: locale]; return dncast(); } + BOOST_CNV_PARAM (base, base::type const) { base_ = arg[ARG:: base]; return dncast(); } + BOOST_CNV_PARAM (adjust, adjust::type const) { adjust_ = arg[ARG:: adjust]; return dncast(); } + BOOST_CNV_PARAM (precision, int const) { precision_ = arg[ARG::precision]; return dncast(); } + BOOST_CNV_PARAM (precision, int) { precision_ = arg[ARG::precision]; return dncast(); } + BOOST_CNV_PARAM (uppercase, bool const) { uppercase_ = arg[ARG::uppercase]; return dncast(); } + BOOST_CNV_PARAM (skipws, bool const) { skipws_ = arg[ARG:: skipws]; return dncast(); } + BOOST_CNV_PARAM (width, int const) { width_ = arg[ARG:: width]; return dncast(); } + BOOST_CNV_PARAM (fill, char const) { fill_ = arg[ARG:: fill]; return dncast(); } + + protected: + + cnvbase() + : + base_ (10), + skipws_ (false), + precision_ (0), + uppercase_ (false), + width_ (0), + fill_ (' '), + adjust_ (boost::cnv::adjust::right) + {} + + template<typename string_type, typename out_type> + void + str_to_(string_type const& str, optional<out_type>& result_out) const + { + cnv::range<string_type const> range (str); + + /**/ if (skipws_) for (; std::isspace(*range.begin()); ++range); + else if (std::isspace(*range.begin())) return; + + dncast().str_to(range, result_out); + } + template<typename in_type, typename string_type> + void + to_str_(in_type value_in, optional<string_type>& result_out) const + { + typedef typename cnv::range<string_type>::value_type char_type; + + char_type buf[bufsize_]; + cnv::range<char_type*> range = dncast().to_str(value_in, buf); + char_type* beg = range.begin(); + char_type* end = range.end(); + + if (beg < end) + { + format_(buf, beg, end); + + result_out = string_type(beg, end); + } + } + + template<typename char_type> + void + format_(char_type* buf, char_type*& beg, char_type*& end) const + { + if (uppercase_) + { + for (char_type* p = beg; p < end; ++p) *p = std::toupper(*p); + } + if (width_) + { + int const num_fillers = (std::max)(0, int(width_ - (end - beg))); + int const num_left = adjust_ == boost::cnv::adjust::left ? 0 + : adjust_ == boost::cnv::adjust::right ? num_fillers + : (num_fillers / 2); + int const num_right = num_fillers - num_left; + int const str_size = end - beg; + bool const move = (beg < buf + num_left) // No room for left fillers + || (buf + bufsize_ < end + num_right); // No room for right fillers + if (move) + { + std::memmove(buf + num_left, beg, str_size * sizeof(char_type)); + beg = buf + num_left; + end = beg + str_size; + } + for (int k = 0; k < num_left; *(--beg) = fill_, ++k); + for (int k = 0; k < num_right; *(end++) = fill_, ++k); + } + } + + derived_type const& dncast () const { return *static_cast<derived_type const*>(this); } + derived_type& dncast () { return *static_cast<derived_type*>(this); } + + // ULONG_MAX(8 bytes) = 18446744073709551615 (20(10) or 32(2) characters) + // double (8 bytes) max is 316 chars + static int const bufsize_ = 1024; + int base_; + bool skipws_; + int precision_; + bool uppercase_; + int width_; + int fill_; + adjust::type adjust_; +// std::locale locale_; +}; + +#undef BOOST_CNV_TO_STRING +#undef BOOST_CNV_STRING_TO +#undef BOOST_CNV_PARAM + +#endif // BOOST_CONVERT_CONVERTER_BASE_HPP |