io.hpp 32 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069
  1. // Boost.Units - A C++ library for zero-overhead dimensional analysis and
  2. // unit/quantity manipulation and conversion
  3. //
  4. // Copyright (C) 2003-2008 Matthias Christian Schabel
  5. // Copyright (C) 2007-2010 Steven Watanabe
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See
  8. // accompanying file LICENSE_1_0.txt or copy at
  9. // http://www.boost.org/LICENSE_1_0.txt)
  10. #ifndef BOOST_UNITS_IO_HPP
  11. #define BOOST_UNITS_IO_HPP
  12. /// \file
  13. /// \brief Stream input and output for rationals, units and quantities.
  14. /// \details Functions and manipulators for output and input of units and quantities.
  15. /// symbol and name format, and engineering and binary autoprefix.
  16. /// Serialization output is also supported.
  17. #include <cassert>
  18. #include <cmath>
  19. #include <string>
  20. #include <iosfwd>
  21. #include <ios>
  22. #include <sstream>
  23. #include <boost/serialization/nvp.hpp>
  24. #include <boost/units/units_fwd.hpp>
  25. #include <boost/units/heterogeneous_system.hpp>
  26. #include <boost/units/make_scaled_unit.hpp>
  27. #include <boost/units/quantity.hpp>
  28. #include <boost/units/scale.hpp>
  29. #include <boost/units/static_rational.hpp>
  30. #include <boost/units/unit.hpp>
  31. #include <boost/units/detail/utility.hpp>
  32. namespace boost {
  33. namespace serialization {
  34. /// Boost Serialization library support for units.
  35. template<class Archive,class System,class Dim>
  36. inline void serialize(Archive& /*ar*/,boost::units::unit<Dim,System>&,const unsigned int /*version*/)
  37. { }
  38. /// Boost Serialization library support for quantities.
  39. template<class Archive,class Unit,class Y>
  40. inline void serialize(Archive& ar,boost::units::quantity<Unit,Y>& q,const unsigned int /*version*/)
  41. {
  42. ar & boost::serialization::make_nvp("value", units::quantity_cast<Y&>(q));
  43. }
  44. } // namespace serialization
  45. namespace units {
  46. // get string representation of arbitrary type.
  47. template<class T> std::string to_string(const T& t)
  48. {
  49. std::stringstream sstr;
  50. sstr << t;
  51. return sstr.str();
  52. }
  53. /// get string representation of integral-valued @c static_rational.
  54. template<integer_type N> std::string to_string(const static_rational<N>&)
  55. {
  56. return to_string(N);
  57. }
  58. /// get string representation of @c static_rational.
  59. template<integer_type N, integer_type D> std::string to_string(const static_rational<N,D>&)
  60. {
  61. return '(' + to_string(N) + '/' + to_string(D) + ')';
  62. }
  63. /// Write @c static_rational to @c std::basic_ostream.
  64. template<class Char, class Traits, integer_type N, integer_type D>
  65. inline std::basic_ostream<Char, Traits>& operator<<(std::basic_ostream<Char, Traits>& os,const static_rational<N,D>& r)
  66. {
  67. os << to_string(r);
  68. return os;
  69. }
  70. /// traits template for unit names.
  71. template<class BaseUnit>
  72. struct base_unit_info
  73. {
  74. /// INTERNAL ONLY
  75. typedef void base_unit_info_primary_template;
  76. /// The full name of the unit (returns BaseUnit::name() by default)
  77. static std::string name()
  78. {
  79. return(BaseUnit::name());
  80. }
  81. /// The symbol for the base unit (Returns BaseUnit::symbol() by default)
  82. static std::string symbol()
  83. {
  84. return(BaseUnit::symbol()); /// \returns BaseUnit::symbol(), for example "m"
  85. }
  86. };
  87. /// \enum format_mode format of output of units, for example "m" or "meter".
  88. enum format_mode
  89. {
  90. symbol_fmt = 0, /// default - reduces unit names to known symbols for both base and derived units.
  91. name_fmt = 1, /// output full unit names for base and derived units, for example "meter".
  92. raw_fmt = 2, /// output only symbols for base units (but not derived units), for example "m".
  93. typename_fmt = 3, /// output demangled typenames (useful only for diagnosis).
  94. fmt_mask = 3 /// Bits used for format.
  95. };
  96. /// \enum autoprefix_mode automatic scaling and prefix (controlled by value of quantity) a, if any,
  97. enum autoprefix_mode
  98. {
  99. autoprefix_none = 0, /// No automatic prefix.
  100. autoprefix_engineering = 4, /// Scale and prefix with 10^3 multiples, 1234.5 m output as 1.2345 km.
  101. autoprefix_binary = 8, /// Scale and prefix with 2^10 (1024) multiples, 1024 as 1 kb.
  102. autoprefix_mask = 12 /// Bits used for autoprefix.
  103. };
  104. namespace detail {
  105. template<bool>
  106. struct xalloc_key_holder
  107. {
  108. static int value;
  109. static bool initialized;
  110. };
  111. template<bool b>
  112. int xalloc_key_holder<b>::value = 0;
  113. template<bool b>
  114. bool xalloc_key_holder<b>::initialized = 0;
  115. struct xalloc_key_initializer_t
  116. {
  117. xalloc_key_initializer_t()
  118. {
  119. if (!xalloc_key_holder<true>::initialized)
  120. {
  121. xalloc_key_holder<true>::value = std::ios_base::xalloc();
  122. xalloc_key_holder<true>::initialized = true;
  123. }
  124. }
  125. };
  126. namespace /**/ {
  127. xalloc_key_initializer_t xalloc_key_initializer;
  128. } // namespace
  129. } // namespace detail
  130. /// returns flags controlling output.
  131. inline long get_flags(std::ios_base& ios, long mask)
  132. {
  133. return(ios.iword(detail::xalloc_key_holder<true>::value) & mask);
  134. }
  135. /// Set new flags controlling output format.
  136. inline void set_flags(std::ios_base& ios, long new_flags, long mask)
  137. {
  138. assert((~mask & new_flags) == 0);
  139. long& flags = ios.iword(detail::xalloc_key_holder<true>::value);
  140. flags = (flags & ~mask) | new_flags;
  141. }
  142. /// returns flags controlling output format.
  143. inline format_mode get_format(std::ios_base& ios)
  144. {
  145. return(static_cast<format_mode>((get_flags)(ios, fmt_mask)));
  146. }
  147. /// Set new flags controlling output format.
  148. inline void set_format(std::ios_base& ios, format_mode new_mode)
  149. {
  150. (set_flags)(ios, new_mode, fmt_mask);
  151. }
  152. /// Set new flags for type_name output format.
  153. inline std::ios_base& typename_format(std::ios_base& ios)
  154. {
  155. (set_format)(ios, typename_fmt);
  156. return(ios);
  157. }
  158. /// set new flag for raw format output, for example "m".
  159. inline std::ios_base& raw_format(std::ios_base& ios)
  160. {
  161. (set_format)(ios, raw_fmt);
  162. return(ios);
  163. }
  164. /// set new format flag for symbol output, for example "m".
  165. inline std::ios_base& symbol_format(std::ios_base& ios)
  166. {
  167. (set_format)(ios, symbol_fmt);
  168. return(ios);
  169. }
  170. /// set new format for name output, for example "meter".
  171. inline std::ios_base& name_format(std::ios_base& ios)
  172. {
  173. (set_format)(ios, name_fmt);
  174. return(ios);
  175. }
  176. /// get autoprefix flags for output.
  177. inline autoprefix_mode get_autoprefix(std::ios_base& ios)
  178. {
  179. return static_cast<autoprefix_mode>((get_flags)(ios, autoprefix_mask));
  180. }
  181. /// Get format for output.
  182. inline void set_autoprefix(std::ios_base& ios, autoprefix_mode new_mode)
  183. {
  184. (set_flags)(ios, new_mode, autoprefix_mask);
  185. }
  186. /// Clear autoprefix flags.
  187. inline std::ios_base& no_prefix(std::ios_base& ios)
  188. {
  189. (set_autoprefix)(ios, autoprefix_none);
  190. return ios;
  191. }
  192. /// Set flag for engineering prefix, so 1234.5 m displays as "1.2345 km".
  193. inline std::ios_base& engineering_prefix(std::ios_base& ios)
  194. {
  195. (set_autoprefix)(ios, autoprefix_engineering);
  196. return ios;
  197. }
  198. /// Set flag for binary prefix, so 1024 byte displays as "1 Kib".
  199. inline std::ios_base& binary_prefix(std::ios_base& ios)
  200. {
  201. (set_autoprefix)(ios, autoprefix_binary);
  202. return ios;
  203. }
  204. namespace detail {
  205. /// \return exponent string like "^1/2".
  206. template<integer_type N, integer_type D>
  207. inline std::string exponent_string(const static_rational<N,D>& r)
  208. {
  209. return '^' + to_string(r);
  210. }
  211. /// \return empty exponent string for integer rational like 2.
  212. template<>
  213. inline std::string exponent_string(const static_rational<1>&)
  214. {
  215. return "";
  216. }
  217. template<class T>
  218. inline std::string base_unit_symbol_string(const T&)
  219. {
  220. return base_unit_info<typename T::tag_type>::symbol() + exponent_string(typename T::value_type());
  221. }
  222. template<class T>
  223. inline std::string base_unit_name_string(const T&)
  224. {
  225. return base_unit_info<typename T::tag_type>::name() + exponent_string(typename T::value_type());
  226. }
  227. // stringify with symbols.
  228. template<int N>
  229. struct symbol_string_impl
  230. {
  231. template<class Begin>
  232. struct apply
  233. {
  234. typedef typename symbol_string_impl<N-1>::template apply<typename Begin::next> next;
  235. static void value(std::string& str)
  236. {
  237. str += base_unit_symbol_string(typename Begin::item()) + ' ';
  238. next::value(str);
  239. }
  240. };
  241. };
  242. template<>
  243. struct symbol_string_impl<1>
  244. {
  245. template<class Begin>
  246. struct apply
  247. {
  248. static void value(std::string& str)
  249. {
  250. str += base_unit_symbol_string(typename Begin::item());
  251. };
  252. };
  253. };
  254. template<>
  255. struct symbol_string_impl<0>
  256. {
  257. template<class Begin>
  258. struct apply
  259. {
  260. static void value(std::string& str)
  261. {
  262. // better shorthand for dimensionless?
  263. str += "dimensionless";
  264. }
  265. };
  266. };
  267. template<int N>
  268. struct scale_symbol_string_impl
  269. {
  270. template<class Begin>
  271. struct apply
  272. {
  273. static void value(std::string& str)
  274. {
  275. str += Begin::item::symbol();
  276. scale_symbol_string_impl<N - 1>::template apply<typename Begin::next>::value(str);
  277. }
  278. };
  279. };
  280. template<>
  281. struct scale_symbol_string_impl<0>
  282. {
  283. template<class Begin>
  284. struct apply
  285. {
  286. static void value(std::string&) { }
  287. };
  288. };
  289. // stringify with names.
  290. template<int N>
  291. struct name_string_impl
  292. {
  293. template<class Begin>
  294. struct apply
  295. {
  296. typedef typename name_string_impl<N-1>::template apply<typename Begin::next> next;
  297. static void value(std::string& str)
  298. {
  299. str += base_unit_name_string(typename Begin::item()) + ' ';
  300. next::value(str);
  301. }
  302. };
  303. };
  304. template<>
  305. struct name_string_impl<1>
  306. {
  307. template<class Begin>
  308. struct apply
  309. {
  310. static void value(std::string& str)
  311. {
  312. str += base_unit_name_string(typename Begin::item());
  313. };
  314. };
  315. };
  316. template<>
  317. struct name_string_impl<0>
  318. {
  319. template<class Begin>
  320. struct apply
  321. {
  322. static void value(std::string& str)
  323. {
  324. str += "dimensionless";
  325. }
  326. };
  327. };
  328. template<int N>
  329. struct scale_name_string_impl
  330. {
  331. template<class Begin>
  332. struct apply
  333. {
  334. static void value(std::string& str)
  335. {
  336. str += Begin::item::name();
  337. scale_name_string_impl<N - 1>::template apply<typename Begin::next>::value(str);
  338. }
  339. };
  340. };
  341. template<>
  342. struct scale_name_string_impl<0>
  343. {
  344. template<class Begin>
  345. struct apply
  346. {
  347. static void value(std::string&) { }
  348. };
  349. };
  350. } // namespace detail
  351. namespace detail {
  352. // These two overloads of symbol_string and name_string will
  353. // will pick up homogeneous_systems. They simply call the
  354. // appropriate function with a heterogeneous_system.
  355. template<class Dimension,class System, class SubFormatter>
  356. inline std::string
  357. to_string_impl(const unit<Dimension,System>&, SubFormatter f)
  358. {
  359. return f(typename reduce_unit<unit<Dimension, System> >::type());
  360. }
  361. /// INTERNAL ONLY
  362. // this overload picks up heterogeneous units that are not scaled.
  363. template<class Dimension,class Units, class Subformatter>
  364. inline std::string
  365. to_string_impl(const unit<Dimension, heterogeneous_system<heterogeneous_system_impl<Units, Dimension, dimensionless_type> > >&, Subformatter f)
  366. {
  367. std::string str;
  368. f.template append_units_to<Units>(str);
  369. return(str);
  370. }
  371. // This overload is a special case for heterogeneous_system which
  372. // is really unitless
  373. /// INTERNAL ONLY
  374. template<class Subformatter>
  375. inline std::string
  376. to_string_impl(const unit<dimensionless_type, heterogeneous_system<heterogeneous_system_impl<dimensionless_type, dimensionless_type, dimensionless_type> > >&, Subformatter)
  377. {
  378. return("dimensionless");
  379. }
  380. // this overload deals with heterogeneous_systems which are unitless
  381. // but scaled.
  382. /// INTERNAL ONLY
  383. template<class Scale, class Subformatter>
  384. inline std::string
  385. to_string_impl(const unit<dimensionless_type, heterogeneous_system<heterogeneous_system_impl<dimensionless_type, dimensionless_type, Scale> > >&, Subformatter f)
  386. {
  387. std::string str;
  388. f.template append_scale_to<Scale>(str);
  389. return(str);
  390. }
  391. // this overload deals with scaled units.
  392. /// INTERNAL ONLY
  393. template<class Dimension,class Units,class Scale, class Subformatter>
  394. inline std::string
  395. to_string_impl(const unit<Dimension, heterogeneous_system<heterogeneous_system_impl<Units, Dimension, Scale> > >&, Subformatter f)
  396. {
  397. std::string str;
  398. f.template append_scale_to<Scale>(str);
  399. std::string without_scale = f(unit<Dimension, heterogeneous_system<heterogeneous_system_impl<Units, Dimension, dimensionless_type> > >());
  400. if (f.is_default_string(without_scale, unit<Dimension, heterogeneous_system<heterogeneous_system_impl<Units, Dimension, dimensionless_type> > >()))
  401. {
  402. str += "(";
  403. str += without_scale;
  404. str += ")";
  405. }
  406. else
  407. {
  408. str += without_scale;
  409. }
  410. return(str);
  411. }
  412. // This overload catches scaled units that have a single base unit
  413. // raised to the first power. It causes si::nano * si::meters to not
  414. // put parentheses around the meters. i.e. nm rather than n(m)
  415. /// INTERNAL ONLY
  416. template<class Dimension,class Unit,class Scale, class Subformatter>
  417. inline std::string
  418. to_string_impl(const unit<Dimension, heterogeneous_system<heterogeneous_system_impl<list<heterogeneous_system_dim<Unit, static_rational<1> >,dimensionless_type>, Dimension, Scale> > >&, Subformatter f)
  419. {
  420. std::string str;
  421. f.template append_scale_to<Scale>(str);
  422. str += f(unit<Dimension, heterogeneous_system<heterogeneous_system_impl<list<heterogeneous_system_dim<Unit, static_rational<1> >, dimensionless_type>, Dimension, dimensionless_type> > >());
  423. return(str);
  424. }
  425. // This overload is necessary to disambiguate.
  426. // it catches units that are unscaled and have a single
  427. // base unit raised to the first power. It is treated the
  428. // same as any other unscaled unit.
  429. /// INTERNAL ONLY
  430. template<class Dimension,class Unit,class Subformatter>
  431. inline std::string
  432. to_string_impl(const unit<Dimension, heterogeneous_system<heterogeneous_system_impl<list<heterogeneous_system_dim<Unit, static_rational<1> >,dimensionless_type>, Dimension, dimensionless_type> > >&, Subformatter f)
  433. {
  434. std::string str;
  435. f.template append_units_to<list<heterogeneous_system_dim<Unit, static_rational<1> >,dimensionless_type> >(str);
  436. return(str);
  437. }
  438. // This overload catches scaled units that have a single scaled base unit
  439. // raised to the first power. It moves that scaling on the base unit
  440. // to the unit level scaling and recurses. By doing this we make sure that
  441. // si::milli * si::kilograms will print g rather than mkg.
  442. //
  443. // This transformation will not be applied if base_unit_info is specialized
  444. // for the scaled base unit.
  445. //
  446. /// INTERNAL ONLY
  447. template<class Dimension,class Unit,class UnitScale, class Scale, class Subformatter>
  448. inline std::string
  449. to_string_impl(
  450. const unit<
  451. Dimension,
  452. heterogeneous_system<
  453. heterogeneous_system_impl<
  454. list<heterogeneous_system_dim<scaled_base_unit<Unit, UnitScale>, static_rational<1> >, dimensionless_type>,
  455. Dimension,
  456. Scale
  457. >
  458. >
  459. >&,
  460. Subformatter f,
  461. typename base_unit_info<scaled_base_unit<Unit, UnitScale> >::base_unit_info_primary_template* = 0)
  462. {
  463. return(f(
  464. unit<
  465. Dimension,
  466. heterogeneous_system<
  467. heterogeneous_system_impl<
  468. list<heterogeneous_system_dim<Unit, static_rational<1> >, dimensionless_type>,
  469. Dimension,
  470. typename mpl::times<Scale, list<scale_list_dim<UnitScale>, dimensionless_type> >::type
  471. >
  472. >
  473. >()));
  474. }
  475. // this overload disambuguates between the overload for an unscaled unit
  476. // and the overload for a scaled base unit raised to the first power.
  477. /// INTERNAL ONLY
  478. template<class Dimension,class Unit,class UnitScale,class Subformatter>
  479. inline std::string
  480. to_string_impl(
  481. const unit<
  482. Dimension,
  483. heterogeneous_system<
  484. heterogeneous_system_impl<
  485. list<heterogeneous_system_dim<scaled_base_unit<Unit, UnitScale>, static_rational<1> >, dimensionless_type>,
  486. Dimension,
  487. dimensionless_type
  488. >
  489. >
  490. >&,
  491. Subformatter f,
  492. typename base_unit_info<scaled_base_unit<Unit, UnitScale> >::base_unit_info_primary_template* = 0)
  493. {
  494. std::string str;
  495. f.template append_units_to<list<heterogeneous_system_dim<scaled_base_unit<Unit, UnitScale>, static_rational<1> >, dimensionless_type> >(str);
  496. return(str);
  497. }
  498. struct format_raw_symbol_impl {
  499. template<class Units>
  500. void append_units_to(std::string& str) {
  501. detail::symbol_string_impl<Units::size::value>::template apply<Units>::value(str);
  502. }
  503. template<class Scale>
  504. void append_scale_to(std::string& str) {
  505. detail::scale_symbol_string_impl<Scale::size::value>::template apply<Scale>::value(str);
  506. }
  507. template<class Unit>
  508. std::string operator()(const Unit& u) {
  509. return(to_string_impl(u, *this));
  510. }
  511. template<class Unit>
  512. bool is_default_string(const std::string&, const Unit&) {
  513. return(true);
  514. }
  515. };
  516. struct format_symbol_impl : format_raw_symbol_impl {
  517. template<class Unit>
  518. std::string operator()(const Unit& u) {
  519. return(symbol_string(u));
  520. }
  521. template<class Unit>
  522. bool is_default_string(const std::string& str, const Unit& u) {
  523. return(str == to_string_impl(u, format_raw_symbol_impl()));
  524. }
  525. };
  526. struct format_raw_name_impl {
  527. template<class Units>
  528. void append_units_to(std::string& str) {
  529. detail::name_string_impl<(Units::size::value)>::template apply<Units>::value(str);
  530. }
  531. template<class Scale>
  532. void append_scale_to(std::string& str) {
  533. detail::scale_name_string_impl<Scale::size::value>::template apply<Scale>::value(str);
  534. }
  535. template<class Unit>
  536. std::string operator()(const Unit& u) {
  537. return(to_string_impl(u, *this));
  538. }
  539. template<class Unit>
  540. bool is_default_string(const std::string&, const Unit&) {
  541. return(true);
  542. }
  543. };
  544. struct format_name_impl : format_raw_name_impl {
  545. template<class Unit>
  546. std::string operator()(const Unit& u) {
  547. return(name_string(u));
  548. }
  549. template<class Unit>
  550. bool is_default_string(const std::string& str, const Unit& u) {
  551. return(str == to_string_impl(u, format_raw_name_impl()));
  552. }
  553. };
  554. template<class Char, class Traits>
  555. inline void do_print(std::basic_ostream<Char, Traits>& os, const std::string& s)
  556. {
  557. os << s.c_str();
  558. }
  559. inline void do_print(std::ostream& os, const std::string& s)
  560. {
  561. os << s;
  562. }
  563. template<class Char, class Traits>
  564. inline void do_print(std::basic_ostream<Char, Traits>& os, const char* s)
  565. {
  566. os << s;
  567. }
  568. // For automatically applying the appropriate prefixes.
  569. }
  570. #ifdef BOOST_UNITS_DOXYGEN
  571. /// ADL customization point for automatic prefixing.
  572. /// Returns a non-negative value. Implemented as std::abs
  573. /// for built-in types.
  574. template<class T>
  575. double autoprefix_norm(const T& arg);
  576. #else
  577. template<class T, bool C = boost::is_arithmetic<T>::value>
  578. struct autoprefix_norm_impl;
  579. template<class T>
  580. struct autoprefix_norm_impl<T, true>
  581. {
  582. typedef double type;
  583. static double call(const T& arg) { return std::abs(arg); }
  584. };
  585. template<class T>
  586. struct autoprefix_norm_impl<T, false>
  587. {
  588. typedef one type;
  589. static one call(const T&) { return one(); }
  590. };
  591. template<class T>
  592. typename autoprefix_norm_impl<T>::type autoprefix_norm(const T& arg)
  593. {
  594. return autoprefix_norm_impl<T>::call(arg);
  595. }
  596. #endif
  597. namespace detail {
  598. template<class End, class Prev, class T, class F>
  599. bool find_matching_scale_impl(End, End, Prev, T, double, F)
  600. {
  601. return false;
  602. }
  603. template<class Begin, class End, class Prev, class T, class F>
  604. bool find_matching_scale_impl(Begin, End end, Prev prev, T t, double x, F f)
  605. {
  606. if(Begin::item::value() > x) {
  607. f(prev, t);
  608. return true;
  609. } else {
  610. return detail::find_matching_scale_impl(
  611. typename Begin::next(),
  612. end,
  613. typename Begin::item(),
  614. t,
  615. x,
  616. f
  617. );
  618. }
  619. }
  620. template<class End, class T, class F>
  621. bool find_matching_scale_i(End, End, T, double, F)
  622. {
  623. return false;
  624. }
  625. template<class Begin, class End, class T, class F>
  626. bool find_matching_scale_i(Begin, End end, T t, double x, F f)
  627. {
  628. if(Begin::item::value() > x) {
  629. return false;
  630. } else {
  631. return detail::find_matching_scale_impl(typename Begin::next(), end, typename Begin::item(), t, x, f);
  632. }
  633. }
  634. template<class Scales, class T, class F>
  635. bool find_matching_scale(T t, double x, F f)
  636. {
  637. return detail::find_matching_scale_i(Scales(), dimensionless_type(), t, x, f);
  638. }
  639. typedef list<scale<10, static_rational<-24> >,
  640. list<scale<10, static_rational<-21> >,
  641. list<scale<10, static_rational<-18> >,
  642. list<scale<10, static_rational<-15> >,
  643. list<scale<10, static_rational<-12> >,
  644. list<scale<10, static_rational<-9> >,
  645. list<scale<10, static_rational<-6> >,
  646. list<scale<10, static_rational<-3> >,
  647. list<scale<10, static_rational<0> >,
  648. list<scale<10, static_rational<3> >,
  649. list<scale<10, static_rational<6> >,
  650. list<scale<10, static_rational<9> >,
  651. list<scale<10, static_rational<12> >,
  652. list<scale<10, static_rational<15> >,
  653. list<scale<10, static_rational<18> >,
  654. list<scale<10, static_rational<21> >,
  655. list<scale<10, static_rational<24> >,
  656. list<scale<10, static_rational<27> >,
  657. dimensionless_type> > > > > > > > > > > > > > > > > > engineering_prefixes;
  658. typedef list<scale<2, static_rational<10> >,
  659. list<scale<2, static_rational<20> >,
  660. list<scale<2, static_rational<30> >,
  661. list<scale<2, static_rational<40> >,
  662. list<scale<2, static_rational<50> >,
  663. list<scale<2, static_rational<60> >,
  664. list<scale<2, static_rational<70> >,
  665. list<scale<2, static_rational<80> >,
  666. list<scale<2, static_rational<90> >,
  667. dimensionless_type> > > > > > > > > binary_prefixes;
  668. template<class Os, class Quantity>
  669. struct print_default_t {
  670. typedef void result_type;
  671. void operator()() const
  672. {
  673. *os << q->value() << ' ' << typename Quantity::unit_type();
  674. }
  675. Os* os;
  676. const Quantity* q;
  677. };
  678. template<class Os, class Quantity>
  679. print_default_t<Os, Quantity> print_default(Os& os, const Quantity& q)
  680. {
  681. print_default_t<Os, Quantity> result = { &os, &q };
  682. return result;
  683. }
  684. template<class Os>
  685. struct print_scale_t {
  686. typedef void result_type;
  687. template<class Prefix, class T>
  688. void operator()(Prefix, const T& t) const
  689. {
  690. *prefixed = true;
  691. *os << t / Prefix::value() << ' ';
  692. switch(units::get_format(*os)) {
  693. case name_fmt: do_print(*os, Prefix::name()); break;
  694. case raw_fmt:
  695. case symbol_fmt: do_print(*os, Prefix::symbol()); break;
  696. case typename_fmt: do_print(*os, units::simplify_typename(Prefix())); *os << ' '; break;
  697. }
  698. }
  699. template<long N, class T>
  700. void operator()(scale<N, static_rational<0> >, const T& t) const
  701. {
  702. *prefixed = false;
  703. *os << t << ' ';
  704. }
  705. Os* os;
  706. bool* prefixed;
  707. };
  708. template<class Os>
  709. print_scale_t<Os> print_scale(Os& os, bool& prefixed)
  710. {
  711. print_scale_t<Os> result = { &os, &prefixed };
  712. return result;
  713. }
  714. // puts parentheses around a unit
  715. /// INTERNAL ONLY
  716. template<class Dimension,class Units,class Scale, class Subformatter>
  717. inline std::string
  718. maybe_parenthesize(const unit<Dimension, heterogeneous_system<heterogeneous_system_impl<Units, Dimension, Scale> > >&, Subformatter f)
  719. {
  720. std::string str;
  721. std::string without_scale = f(unit<Dimension, heterogeneous_system<heterogeneous_system_impl<Units, Dimension, dimensionless_type> > >());
  722. if (f.is_default_string(without_scale, unit<Dimension, heterogeneous_system<heterogeneous_system_impl<Units, Dimension, dimensionless_type> > >()))
  723. {
  724. str += "(";
  725. str += without_scale;
  726. str += ")";
  727. }
  728. else
  729. {
  730. str += without_scale;
  731. }
  732. return(str);
  733. }
  734. // This overload catches scaled units that have a single base unit
  735. // raised to the first power. It causes si::nano * si::meters to not
  736. // put parentheses around the meters. i.e. nm rather than n(m)
  737. /// INTERNAL ONLY
  738. template<class Dimension,class Unit,class Scale, class Subformatter>
  739. inline std::string
  740. maybe_parenthesize(const unit<Dimension, heterogeneous_system<heterogeneous_system_impl<list<heterogeneous_system_dim<Unit, static_rational<1> >,dimensionless_type>, Dimension, Scale> > >&, Subformatter f)
  741. {
  742. return f(unit<Dimension, heterogeneous_system<heterogeneous_system_impl<list<heterogeneous_system_dim<Unit, static_rational<1> >, dimensionless_type>, Dimension, dimensionless_type> > >());
  743. }
  744. template<class Prefixes, class CharT, class Traits, class Unit, class T, class F>
  745. void do_print_prefixed_impl(std::basic_ostream<CharT, Traits>& os, const quantity<Unit, T>& q, F default_)
  746. {
  747. bool prefixed;
  748. if(detail::find_matching_scale<Prefixes>(q.value(), autoprefix_norm(q.value()), detail::print_scale(os, prefixed))) {
  749. if(prefixed) {
  750. switch(units::get_format(os)) {
  751. case symbol_fmt: do_print(os, maybe_parenthesize(Unit(), format_symbol_impl())); break;
  752. case raw_fmt: do_print(os, maybe_parenthesize(Unit(), format_raw_symbol_impl())); break;
  753. case name_fmt: do_print(os, maybe_parenthesize(Unit(), format_name_impl())); break;
  754. case typename_fmt: do_print(os, simplify_typename(Unit())); break;
  755. }
  756. } else {
  757. os << Unit();
  758. }
  759. } else {
  760. default_();
  761. }
  762. }
  763. // Handle units like si::kilograms that have a scale embedded in the
  764. // base unit. This overload is disabled if the scaled base unit has
  765. // a user-defined string representation.
  766. template<class Prefixes, class CharT, class Traits, class Dimension, class BaseUnit, class BaseScale, class Scale, class T>
  767. typename base_unit_info<
  768. scaled_base_unit<BaseUnit, Scale>
  769. >::base_unit_info_primary_template
  770. do_print_prefixed(
  771. std::basic_ostream<CharT, Traits>& os,
  772. const quantity<
  773. unit<
  774. Dimension,
  775. heterogeneous_system<
  776. heterogeneous_system_impl<
  777. list<
  778. heterogeneous_system_dim<
  779. scaled_base_unit<BaseUnit, BaseScale>,
  780. static_rational<1>
  781. >,
  782. dimensionless_type
  783. >,
  784. Dimension,
  785. Scale
  786. >
  787. >
  788. >,
  789. T
  790. >& q)
  791. {
  792. quantity<
  793. unit<
  794. Dimension,
  795. heterogeneous_system<
  796. heterogeneous_system_impl<
  797. list<
  798. heterogeneous_system_dim<BaseUnit, static_rational<1> >,
  799. dimensionless_type
  800. >,
  801. Dimension,
  802. dimensionless_type
  803. >
  804. >
  805. >,
  806. T
  807. > unscaled(q);
  808. detail::do_print_prefixed_impl<Prefixes>(os, unscaled, detail::print_default(os, q));
  809. }
  810. template<class Prefixes, class CharT, class Traits, class Dimension, class L, class Scale, class T>
  811. void do_print_prefixed(
  812. std::basic_ostream<CharT, Traits>& os,
  813. const quantity<
  814. unit<
  815. Dimension,
  816. heterogeneous_system<
  817. heterogeneous_system_impl<
  818. L,
  819. Dimension,
  820. Scale
  821. >
  822. >
  823. >,
  824. T
  825. >& q)
  826. {
  827. quantity<
  828. unit<
  829. Dimension,
  830. heterogeneous_system<
  831. heterogeneous_system_impl<
  832. L,
  833. Dimension,
  834. dimensionless_type
  835. >
  836. >
  837. >,
  838. T
  839. > unscaled(q);
  840. detail::do_print_prefixed_impl<Prefixes>(os, unscaled, detail::print_default(os, q));
  841. }
  842. template<class Prefixes, class CharT, class Traits, class Dimension, class System, class T>
  843. void do_print_prefixed(std::basic_ostream<CharT, Traits>& os, const quantity<unit<Dimension, System>, T>& q)
  844. {
  845. detail::do_print_prefixed<Prefixes>(os, quantity<unit<Dimension, typename make_heterogeneous_system<Dimension, System>::type>, T>(q));
  846. }
  847. template<class Prefixes, class CharT, class Traits, class Unit, class T>
  848. void do_print_prefixed(std::basic_ostream<CharT, Traits>& os, const quantity<Unit, T>& q)
  849. {
  850. detail::print_default(os, q)();
  851. }
  852. template<class Prefixes, class CharT, class Traits, class Unit, class T>
  853. void maybe_print_prefixed(std::basic_ostream<CharT, Traits>& os, const quantity<Unit, T>& q, mpl::true_)
  854. {
  855. detail::do_print_prefixed<Prefixes>(os, q);
  856. }
  857. template<class Prefixes, class CharT, class Traits, class Unit, class T>
  858. void maybe_print_prefixed(std::basic_ostream<CharT, Traits>& os, const quantity<Unit, T>& q, mpl::false_)
  859. {
  860. detail::print_default(os, q)();
  861. }
  862. inline mpl::true_ test_norm(double) { return mpl::true_(); }
  863. inline mpl::false_ test_norm(one) { return mpl::false_(); }
  864. } // namespace detail
  865. template<class Dimension,class System>
  866. inline std::string
  867. typename_string(const unit<Dimension, System>&)
  868. {
  869. return simplify_typename(typename reduce_unit< unit<Dimension,System> >::type());
  870. }
  871. template<class Dimension,class System>
  872. inline std::string
  873. symbol_string(const unit<Dimension, System>&)
  874. {
  875. return detail::to_string_impl(unit<Dimension,System>(), detail::format_symbol_impl());
  876. }
  877. template<class Dimension,class System>
  878. inline std::string
  879. name_string(const unit<Dimension, System>&)
  880. {
  881. return detail::to_string_impl(unit<Dimension,System>(), detail::format_name_impl());
  882. }
  883. /// Print a @c unit as a list of base units and their exponents.
  884. ///
  885. /// for @c symbol_format outputs e.g. "m s^-1" or "J".
  886. /// for @c name_format outputs e.g. "meter second^-1" or "joule".
  887. /// for @c raw_format outputs e.g. "m s^-1" or "meter kilogram^2 second^-2".
  888. /// for @c typename_format outputs the typename itself (currently demangled only on GCC).
  889. template<class Char, class Traits, class Dimension, class System>
  890. inline std::basic_ostream<Char, Traits>& operator<<(std::basic_ostream<Char, Traits>& os, const unit<Dimension, System>& u)
  891. {
  892. if (units::get_format(os) == typename_fmt)
  893. {
  894. detail::do_print(os, typename_string(u));
  895. }
  896. else if (units::get_format(os) == raw_fmt)
  897. {
  898. detail::do_print(os, detail::to_string_impl(u, detail::format_raw_symbol_impl()));
  899. }
  900. else if (units::get_format(os) == symbol_fmt)
  901. {
  902. detail::do_print(os, symbol_string(u));
  903. }
  904. else if (units::get_format(os) == name_fmt)
  905. {
  906. detail::do_print(os, name_string(u));
  907. }
  908. else
  909. {
  910. assert(!"The format mode must be one of: typename_format, raw_format, name_format, symbol_format");
  911. }
  912. return(os);
  913. }
  914. /// \brief Print a @c quantity.
  915. /// \details Prints the value followed by the unit.
  916. /// If the engineering_prefix, or binary_prefix is set,
  917. /// tries to scale the value appropriately.
  918. /// For example, it might print 12.345 km instead of 12345 m.
  919. /// (Note does @b not attempt to automatically scale scalars like double, float...)
  920. template<class Char, class Traits, class Unit, class T>
  921. inline std::basic_ostream<Char, Traits>& operator<<(std::basic_ostream<Char, Traits>& os, const quantity<Unit, T>& q)
  922. {
  923. if (units::get_autoprefix(os) == autoprefix_none)
  924. {
  925. os << q.value() << ' ' << Unit();
  926. }
  927. else if (units::get_autoprefix(os) == autoprefix_engineering)
  928. {
  929. detail::maybe_print_prefixed<detail::engineering_prefixes>(os, q, detail::test_norm(autoprefix_norm(q.value())));
  930. }
  931. else if (units::get_autoprefix(os) == autoprefix_binary)
  932. {
  933. detail::maybe_print_prefixed<detail::binary_prefixes>(os, q, detail::test_norm(autoprefix_norm(q.value())));
  934. }
  935. else
  936. {
  937. assert(!"Autoprefixing must be one of: no_prefix, engineering_prefix, binary_prefix");
  938. }
  939. return(os);
  940. }
  941. } // namespace units
  942. } // namespace boost
  943. #endif