format.hpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515
  1. //
  2. // Copyright (c) 2009-2011 Artyom Beilis (Tonkikh)
  3. //
  4. // Distributed under the Boost Software License, Version 1.0. (See
  5. // accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt)
  7. //
  8. #ifndef BOOST_LOCALE_FORMAT_HPP_INCLUDED
  9. #define BOOST_LOCALE_FORMAT_HPP_INCLUDED
  10. #include <boost/locale/config.hpp>
  11. #ifdef BOOST_MSVC
  12. # pragma warning(push)
  13. # pragma warning(disable : 4275 4251 4231 4660)
  14. #endif
  15. #include <boost/locale/message.hpp>
  16. #include <boost/locale/formatting.hpp>
  17. #include <sstream>
  18. namespace boost {
  19. namespace locale {
  20. ///
  21. /// \defgroup format Format
  22. ///
  23. /// This module provides printf like functionality integrated into iostreams and suitable for localization
  24. ///
  25. /// @{
  26. ///
  27. /// \cond INTERNAL
  28. namespace details {
  29. template<typename CharType>
  30. struct formattible {
  31. typedef std::basic_ostream<CharType> stream_type;
  32. typedef void (*writer_type)(stream_type &output,void const *ptr);
  33. formattible() :
  34. pointer_(0),
  35. writer_(&formattible::void_write)
  36. {
  37. }
  38. formattible(formattible const &other) :
  39. pointer_(other.pointer_),
  40. writer_(other.writer_)
  41. {
  42. }
  43. formattible const &operator=(formattible const &other)
  44. {
  45. if(this != &other) {
  46. pointer_=other.pointer_;
  47. writer_=other.writer_;
  48. }
  49. return *this;
  50. }
  51. template<typename Type>
  52. formattible(Type const &value)
  53. {
  54. pointer_ = static_cast<void const *>(&value);
  55. writer_ = &write<Type>;
  56. }
  57. template<typename Type>
  58. formattible const &operator=(Type const &other)
  59. {
  60. *this = formattible(other);
  61. return *this;
  62. }
  63. friend stream_type &operator<<(stream_type &out,formattible const &fmt)
  64. {
  65. fmt.writer_(out,fmt.pointer_);
  66. return out;
  67. }
  68. private:
  69. static void void_write(stream_type &output,void const * /*ptr*/)
  70. {
  71. CharType empty_string[1]={0};
  72. output<<empty_string;
  73. }
  74. template<typename Type>
  75. static void write(stream_type &output,void const *ptr)
  76. {
  77. output << *static_cast<Type const *>(ptr);
  78. }
  79. void const *pointer_;
  80. writer_type writer_;
  81. }; // formattible
  82. class BOOST_LOCALE_DECL format_parser {
  83. public:
  84. format_parser(std::ios_base &ios,void *,void (*imbuer)(void *,std::locale const &));
  85. ~format_parser();
  86. unsigned get_position();
  87. void set_one_flag(std::string const &key,std::string const &value);
  88. template<typename CharType>
  89. void set_flag_with_str(std::string const &key,std::basic_string<CharType> const &value)
  90. {
  91. if(key=="ftime" || key=="strftime") {
  92. as::strftime(ios_);
  93. ios_info::get(ios_).date_time_pattern(value);
  94. }
  95. }
  96. void restore();
  97. private:
  98. void imbue(std::locale const &);
  99. format_parser(format_parser const &);
  100. void operator=(format_parser const &);
  101. std::ios_base &ios_;
  102. struct data;
  103. std::auto_ptr<data> d;
  104. };
  105. }
  106. /// \endcond
  107. ///
  108. /// \brief a printf like class that allows type-safe and locale aware message formatting
  109. ///
  110. /// This class creates a formatted message similar to printf or boost::format and receives
  111. /// formatted entries via operator %.
  112. ///
  113. /// For example
  114. /// \code
  115. /// cout << format("Hello {1}, you are {2} years old") % name % age << endl;
  116. /// \endcode
  117. ///
  118. /// Formatting is enclosed between curly brackets \c { \c } and defined by a comma separated list of flags in the format key[=value]
  119. /// value may also be text included between single quotes \c ' that is used for special purposes where inclusion of non-ASCII
  120. /// text is allowed
  121. ///
  122. /// Including of literal \c { and \c } is possible by specifying double brackets \c {{ and \c }} accordingly.
  123. ///
  124. ///
  125. /// For example:
  126. ///
  127. /// \code
  128. /// cout << format("The height of water at {1,time} is {2,num=fixed,precision=3}") % time % height;
  129. /// \endcode
  130. ///
  131. /// The special key -- a number without a value defines the position of an input parameter.
  132. /// List of keys:
  133. /// - \c [0-9]+ -- digits, the index of a formatted parameter -- mandatory key.
  134. /// - \c num or \c number -- format a number. Optional values are:
  135. /// - \c hex -- display hexadecimal number
  136. /// - \c oct -- display in octal format
  137. /// - \c sci or \c scientific -- display in scientific format
  138. /// - \c fix or \c fixed -- display in fixed format
  139. /// .
  140. /// For example \c number=sci
  141. /// - \c cur or \c currency -- format currency. Optional values are:
  142. ///
  143. /// - \c iso -- display using ISO currency symbol.
  144. /// - \c nat or \c national -- display using national currency symbol.
  145. /// .
  146. /// - \c per or \c percent -- format percent value.
  147. /// - \c date, \c time , \c datetime or \c dt -- format date, time or date and time. Optional values are:
  148. /// - \c s or \c short -- display in short format
  149. /// - \c m or \c medium -- display in medium format.
  150. /// - \c l or \c long -- display in long format.
  151. /// - \c f or \c full -- display in full format.
  152. /// .
  153. /// - \c ftime with string (quoted) parameter -- display as with \c strftime see, \c as::ftime manipulator
  154. /// - \c spell or \c spellout -- spell the number.
  155. /// - \c ord or \c ordinal -- format ordinal number (1st, 2nd... etc)
  156. /// - \c left or \c < -- align to left.
  157. /// - \c right or \c > -- align to right.
  158. /// - \c width or \c w -- set field width (requires parameter).
  159. /// - \c precision or \c p -- set precision (requires parameter).
  160. /// - \c locale -- with parameter -- switch locale for current operation. This command generates locale
  161. /// with formatting facets giving more fine grained control of formatting. For example:
  162. /// \code
  163. /// cout << format("Today {1,date} ({1,date,locale=he_IL.UTF-8@calendar=hebrew,date} Hebrew Date)") % date;
  164. /// \endcode
  165. /// - \c timezone or \c tz -- the name of the timezone to display the time in. For example:\n
  166. /// \code
  167. /// cout << format("Time is: Local {1,time}, ({1,time,tz=EET} Eastern European Time)") % date;
  168. /// \endcode
  169. /// - \c local - display the time in local time
  170. /// - \c gmt - display the time in UTC time scale
  171. /// \code
  172. /// cout << format("Local time is: {1,time,local}, universal time is {1,time,gmt}") % time;
  173. /// \endcode
  174. ///
  175. ///
  176. /// Invalid formatting strings are slightly ignored. This would prevent from translator
  177. /// to crash the program in unexpected location.
  178. ///
  179. template<typename CharType>
  180. class basic_format {
  181. public:
  182. typedef CharType char_type; ///< Underlying character type
  183. typedef basic_message<char_type> message_type; ///< The translation message type
  184. /// \cond INTERNAL
  185. typedef details::formattible<CharType> formattible_type;
  186. /// \endcond
  187. typedef std::basic_string<CharType> string_type; ///< string type for this type of character
  188. typedef std::basic_ostream<CharType> stream_type; ///< output stream type for this type of character
  189. ///
  190. /// Create a format class for \a format_string
  191. ///
  192. basic_format(string_type format_string) :
  193. format_(format_string),
  194. translate_(false),
  195. parameters_count_(0)
  196. {
  197. }
  198. ///
  199. /// Create a format class using message \a trans. The message if translated first according
  200. /// to the rules of target locale and then interpreted as format string
  201. ///
  202. basic_format(message_type const &trans) :
  203. message_(trans),
  204. translate_(true),
  205. parameters_count_(0)
  206. {
  207. }
  208. ///
  209. /// Add new parameter to format list. The object should be a type
  210. /// with defined expression out << object where \c out is \c std::basic_ostream.
  211. ///
  212. template<typename Formattible>
  213. basic_format &operator % (Formattible const &object)
  214. {
  215. add(formattible_type(object));
  216. return *this;
  217. }
  218. ///
  219. /// Format a string using a locale \a loc
  220. ///
  221. string_type str(std::locale const &loc = std::locale()) const
  222. {
  223. std::basic_ostringstream<CharType> buffer;
  224. buffer.imbue(loc);
  225. write(buffer);
  226. return buffer.str();
  227. }
  228. ///
  229. /// write a formatted string to output stream \a out using out's locale
  230. ///
  231. void write(stream_type &out) const
  232. {
  233. string_type format;
  234. if(translate_)
  235. format = message_.str(out.getloc(),ios_info::get(out).domain_id());
  236. else
  237. format = format_;
  238. format_output(out,format);
  239. }
  240. private:
  241. class format_guard {
  242. public:
  243. format_guard(details::format_parser &fmt) :
  244. fmt_(&fmt),
  245. restored_(false)
  246. {
  247. }
  248. void restore()
  249. {
  250. if(restored_)
  251. return;
  252. fmt_->restore();
  253. restored_ = true;
  254. }
  255. ~format_guard()
  256. {
  257. try {
  258. restore();
  259. }
  260. catch(...) {
  261. }
  262. }
  263. private:
  264. details::format_parser *fmt_;
  265. bool restored_;
  266. };
  267. void format_output(stream_type &out,string_type const &sformat) const
  268. {
  269. char_type obrk='{';
  270. char_type cbrk='}';
  271. char_type eq='=';
  272. char_type comma=',';
  273. char_type quote='\'';
  274. size_t pos = 0;
  275. size_t size=sformat.size();
  276. CharType const *format=sformat.c_str();
  277. while(format[pos]!=0) {
  278. if(format[pos] != obrk) {
  279. if(format[pos]==cbrk && format[pos+1]==cbrk) {
  280. out << cbrk;
  281. pos+=2;
  282. }
  283. else {
  284. out<<format[pos];
  285. pos++;
  286. }
  287. continue;
  288. }
  289. if(pos+1 < size && format[pos+1]==obrk) {
  290. out << obrk;
  291. pos+=2;
  292. continue;
  293. }
  294. pos++;
  295. details::format_parser fmt(out,static_cast<void *>(&out),&basic_format::imbue_locale);
  296. format_guard guard(fmt);
  297. while(pos < size) {
  298. std::string key;
  299. std::string svalue;
  300. string_type value;
  301. bool use_svalue = true;
  302. for(;format[pos];pos++) {
  303. char_type c=format[pos];
  304. if(c==comma || c==eq || c==cbrk)
  305. break;
  306. else {
  307. key+=static_cast<char>(c);
  308. }
  309. }
  310. if(format[pos]==eq) {
  311. pos++;
  312. if(format[pos]==quote) {
  313. pos++;
  314. use_svalue = false;
  315. while(format[pos]) {
  316. if(format[pos]==quote) {
  317. if(format[pos+1]==quote) {
  318. value+=quote;
  319. pos+=2;
  320. }
  321. else {
  322. pos++;
  323. break;
  324. }
  325. }
  326. else {
  327. value+=format[pos];
  328. pos++;
  329. }
  330. }
  331. }
  332. else {
  333. char_type c;
  334. while((c=format[pos])!=0 && c!=comma && c!=cbrk) {
  335. svalue+=static_cast<char>(c);
  336. pos++;
  337. }
  338. }
  339. }
  340. if(use_svalue) {
  341. fmt.set_one_flag(key,svalue);
  342. }
  343. else
  344. fmt.set_flag_with_str(key,value);
  345. if(format[pos]==comma) {
  346. pos++;
  347. continue;
  348. }
  349. else if(format[pos]==cbrk) {
  350. unsigned position = fmt.get_position();
  351. out << get(position);
  352. guard.restore();
  353. pos++;
  354. break;
  355. }
  356. else {
  357. guard.restore();
  358. break;
  359. }
  360. }
  361. }
  362. }
  363. //
  364. // Non-copyable
  365. //
  366. basic_format(basic_format const &other);
  367. void operator=(basic_format const &other);
  368. void add(formattible_type const &param)
  369. {
  370. if(parameters_count_ >= base_params_)
  371. ext_params_.push_back(param);
  372. else
  373. parameters_[parameters_count_] = param;
  374. parameters_count_++;
  375. }
  376. formattible_type get(unsigned id) const
  377. {
  378. if(id >= parameters_count_)
  379. return formattible_type();
  380. else if(id >= base_params_)
  381. return ext_params_[id - base_params_];
  382. else
  383. return parameters_[id];
  384. }
  385. static void imbue_locale(void *ptr,std::locale const &l)
  386. {
  387. reinterpret_cast<stream_type *>(ptr)->imbue(l);
  388. }
  389. static unsigned const base_params_ = 8;
  390. message_type message_;
  391. string_type format_;
  392. bool translate_;
  393. formattible_type parameters_[base_params_];
  394. unsigned parameters_count_;
  395. std::vector<formattible_type> ext_params_;
  396. };
  397. ///
  398. /// Write formatted message to stream.
  399. ///
  400. /// This operator actually causes actual text formatting. It uses the locale of \a out stream
  401. ///
  402. template<typename CharType>
  403. std::basic_ostream<CharType> &operator<<(std::basic_ostream<CharType> &out,basic_format<CharType> const &fmt)
  404. {
  405. fmt.write(out);
  406. return out;
  407. }
  408. ///
  409. /// Definition of char based format
  410. ///
  411. typedef basic_format<char> format;
  412. ///
  413. /// Definition of wchar_t based format
  414. ///
  415. typedef basic_format<wchar_t> wformat;
  416. #ifdef BOOST_LOCALE_ENABLE_CHAR16_T
  417. ///
  418. /// Definition of char16_t based format
  419. ///
  420. typedef basic_format<char16_t> u16format;
  421. #endif
  422. #ifdef BOOST_LOCALE_ENABLE_CHAR32_T
  423. ///
  424. /// Definition of char32_t based format
  425. ///
  426. typedef basic_format<char32_t> u32format;
  427. #endif
  428. ///
  429. /// @}
  430. ///
  431. }
  432. }
  433. #ifdef BOOST_MSVC
  434. #pragma warning(pop)
  435. #endif
  436. #endif
  437. ///
  438. /// \example hello.cpp
  439. ///
  440. /// Basic example of using various functions provided by this library
  441. ///
  442. /// \example whello.cpp
  443. ///
  444. /// Basic example of using various functions with wide strings provided by this library
  445. ///
  446. ///
  447. // vim: tabstop=4 expandtab shiftwidth=4 softtabstop=4