| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2012-12-12 07:44:41 -08:00
										 |  |  |  String formatting library for C++ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  Copyright (c) 2012, Victor Zverovich | 
					
						
							|  |  |  |  All rights reserved. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  modification, are permitted provided that the following conditions are met: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  1. Redistributions of source code must retain the above copyright notice, this | 
					
						
							|  |  |  |     list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  2. Redistributions in binary form must reproduce the above copyright notice, | 
					
						
							|  |  |  |     this list of conditions and the following disclaimer in the documentation | 
					
						
							|  |  |  |     and/or other materials provided with the distribution. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND | 
					
						
							|  |  |  |  ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | 
					
						
							|  |  |  |  WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
					
						
							|  |  |  |  DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR | 
					
						
							|  |  |  |  ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | 
					
						
							|  |  |  |  (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | 
					
						
							|  |  |  |  LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | 
					
						
							|  |  |  |  ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
					
						
							|  |  |  |  (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | 
					
						
							|  |  |  |  SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef FORMAT_H_
 | 
					
						
							|  |  |  | #define FORMAT_H_
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | #include <stdint.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | #include <cassert>
 | 
					
						
							|  |  |  | #include <climits>
 | 
					
						
							|  |  |  | #include <cmath>
 | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | #include <cstddef>
 | 
					
						
							| 
									
										
										
										
											2012-12-18 15:50:14 -08:00
										 |  |  | #include <cstdio>
 | 
					
						
							| 
									
										
										
										
											2012-12-18 15:39:42 -08:00
										 |  |  | #include <cstring>
 | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | #include <stdexcept>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2012-12-09 09:03:47 -08:00
										 |  |  | #include <sstream>
 | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-21 11:42:25 -08:00
										 |  |  | namespace fmt { | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | namespace internal { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  | // A simple array for POD types with the first SIZE elements stored in
 | 
					
						
							|  |  |  | // the object itself. It supports a subset of std::vector's operations.
 | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  | template <typename T, std::size_t SIZE> | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  | class Array { | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |  private: | 
					
						
							|  |  |  |   std::size_t size_; | 
					
						
							|  |  |  |   std::size_t capacity_; | 
					
						
							|  |  |  |   T *ptr_; | 
					
						
							|  |  |  |   T data_[SIZE]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 18:08:04 -08:00
										 |  |  |   void Grow(std::size_t size); | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Do not implement!
 | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  |   Array(const Array &); | 
					
						
							|  |  |  |   void operator=(const Array &); | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  |   Array() : size_(0), capacity_(SIZE), ptr_(data_) {} | 
					
						
							|  |  |  |   ~Array() { | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |     if (ptr_ != data_) delete [] ptr_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  |   // Returns the size of this array.
 | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |   std::size_t size() const { return size_; } | 
					
						
							| 
									
										
										
										
											2012-12-10 20:37:35 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  |   // Returns the capacity of this array.
 | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |   std::size_t capacity() const { return capacity_; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  |   // Resizes the array. If T is a POD type new elements are not initialized.
 | 
					
						
							| 
									
										
										
										
											2012-12-10 18:08:04 -08:00
										 |  |  |   void resize(std::size_t new_size) { | 
					
						
							|  |  |  |     if (new_size > capacity_) | 
					
						
							|  |  |  |       Grow(new_size); | 
					
						
							|  |  |  |     size_ = new_size; | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   void reserve(std::size_t capacity) { | 
					
						
							| 
									
										
										
										
											2012-12-10 18:08:04 -08:00
										 |  |  |     if (capacity > capacity_) | 
					
						
							|  |  |  |       Grow(capacity); | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 20:37:35 -08:00
										 |  |  |   void clear() { size_ = 0; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |   void push_back(const T &value) { | 
					
						
							|  |  |  |     if (size_ == capacity_) | 
					
						
							|  |  |  |       Grow(size_ + 1); | 
					
						
							|  |  |  |     ptr_[size_++] = value; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  |   // Appends data to the end of the array.
 | 
					
						
							| 
									
										
										
										
											2012-12-10 18:08:04 -08:00
										 |  |  |   void append(const T *begin, const T *end); | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   T &operator[](std::size_t index) { return ptr_[index]; } | 
					
						
							|  |  |  |   const T &operator[](std::size_t index) const { return ptr_[index]; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 18:08:04 -08:00
										 |  |  | template <typename T, std::size_t SIZE> | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  | void Array<T, SIZE>::Grow(std::size_t size) { | 
					
						
							| 
									
										
										
										
											2012-12-10 18:08:04 -08:00
										 |  |  |   capacity_ = std::max(size, capacity_ + capacity_ / 2); | 
					
						
							|  |  |  |   T *p = new T[capacity_]; | 
					
						
							|  |  |  |   std::copy(ptr_, ptr_ + size_, p); | 
					
						
							|  |  |  |   if (ptr_ != data_) | 
					
						
							|  |  |  |     delete [] ptr_; | 
					
						
							|  |  |  |   ptr_ = p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T, std::size_t SIZE> | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  | void Array<T, SIZE>::append(const T *begin, const T *end) { | 
					
						
							| 
									
										
										
										
											2012-12-10 18:08:04 -08:00
										 |  |  |   std::ptrdiff_t num_elements = end - begin; | 
					
						
							|  |  |  |   if (size_ + num_elements > capacity_) | 
					
						
							|  |  |  |     Grow(num_elements); | 
					
						
							|  |  |  |   std::copy(begin, end, ptr_ + size_); | 
					
						
							|  |  |  |   size_ += num_elements; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | // Information about an integer type.
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  | // IntTraits is not specialized for integer types smaller than int,
 | 
					
						
							|  |  |  | // since these are promoted to int.
 | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | template <typename T> | 
					
						
							| 
									
										
										
										
											2013-01-13 09:41:14 -08:00
										 |  |  | struct IntTraits { | 
					
						
							|  |  |  |   typedef T UnsignedType; | 
					
						
							|  |  |  |   static bool IsNegative(T) { return false; } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | template <typename T, typename UnsignedT> | 
					
						
							|  |  |  | struct SignedIntTraits { | 
					
						
							|  |  |  |   typedef UnsignedT UnsignedType; | 
					
						
							|  |  |  |   static bool IsNegative(T value) { return value < 0; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | template <> | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  | struct IntTraits<int> : SignedIntTraits<int, unsigned> {}; | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  | template <> | 
					
						
							| 
									
										
										
										
											2013-01-13 09:29:37 -08:00
										 |  |  | struct IntTraits<long> : SignedIntTraits<long, unsigned long> {}; | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  | template <typename T> | 
					
						
							|  |  |  | struct IsLongDouble { enum {VALUE = 0}; }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | struct IsLongDouble<long double> { enum {VALUE = 1}; }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern const char DIGITS[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ReportUnknownType(char code, const char *type); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | // Returns the number of decimal digits in n. Trailing zeros are not counted
 | 
					
						
							|  |  |  | // except for n == 0 in which case CountDigits returns 1.
 | 
					
						
							|  |  |  | inline unsigned CountDigits(uint64_t n) { | 
					
						
							|  |  |  |   unsigned count = 1; | 
					
						
							|  |  |  |   for (;;) { | 
					
						
							|  |  |  |     // Integer division is slow so do it for a group of four digits instead
 | 
					
						
							|  |  |  |     // of for every digit. The idea comes from the talk by Alexandrescu
 | 
					
						
							|  |  |  |     // "Three Optimization Tips for C++". See speed-test for a comparison.
 | 
					
						
							|  |  |  |     if (n < 10) return count; | 
					
						
							|  |  |  |     if (n < 100) return count + 1; | 
					
						
							|  |  |  |     if (n < 1000) return count + 2; | 
					
						
							|  |  |  |     if (n < 10000) return count + 3; | 
					
						
							|  |  |  |     n /= 10000u; | 
					
						
							|  |  |  |     count += 4; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef _MSC_VER
 | 
					
						
							| 
									
										
										
										
											2013-02-05 08:11:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | inline int SignBit(double value) { | 
					
						
							|  |  |  |   // When compiled in C++11 mode signbit is no longer a macro but a function
 | 
					
						
							|  |  |  |   // defined in namespace std and the macro is undefined.
 | 
					
						
							|  |  |  |   using namespace std; | 
					
						
							|  |  |  |   return signbit(value); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-02-05 08:11:04 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | inline int IsInf(double x) { | 
					
						
							|  |  |  |   using namespace std; | 
					
						
							| 
									
										
										
										
											2013-02-21 16:53:58 -08:00
										 |  |  |   return ::isinf(x); | 
					
						
							| 
									
										
										
										
											2013-02-05 08:11:04 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2013-02-05 08:11:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | inline int SignBit(double value) { | 
					
						
							|  |  |  |   if (value < 0) return 1; | 
					
						
							|  |  |  |   if (value == value) return 0; | 
					
						
							|  |  |  |   int dec = 0, sign = 0; | 
					
						
							|  |  |  |   _ecvt(value, 0, &dec, &sign); | 
					
						
							|  |  |  |   return sign; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-02-05 08:11:04 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | inline int IsInf(double x) { return !_finite(x); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:43:15 -08:00
										 |  |  | # undef snprintf
 | 
					
						
							|  |  |  | # define snprintf _snprintf
 | 
					
						
							| 
									
										
										
										
											2013-02-05 08:11:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | class ArgInserter; | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  | class FormatterProxy; | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-04 07:04:35 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |   \rst | 
					
						
							|  |  |  |   A string reference. It can be constructed from a C string, ``std::string`` | 
					
						
							|  |  |  |   or as a result of a formatting operation. It is most useful as a parameter | 
					
						
							|  |  |  |   type to allow passing different types of strings in a function, for example:: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     TempFormatter<> Format(StringRef format); | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Format("{}") << 42; | 
					
						
							|  |  |  |     Format(std::string("{}")) << 42; | 
					
						
							|  |  |  |     Format(Format("{{}}")) << 42; | 
					
						
							| 
									
										
										
										
											2013-01-04 07:04:35 -08:00
										 |  |  |   \endrst | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2012-12-25 13:25:14 -08:00
										 |  |  | class StringRef { | 
					
						
							|  |  |  |  private: | 
					
						
							|  |  |  |   const char *data_; | 
					
						
							|  |  |  |   mutable std::size_t size_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Constructs a string reference object from a C string and a size. | 
					
						
							|  |  |  |     If `size` is zero, which is the default, the size is computed with | 
					
						
							|  |  |  |     `strlen`. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2012-12-25 13:25:14 -08:00
										 |  |  |   StringRef(const char *s, std::size_t size = 0) : data_(s), size_(size) {} | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |     Constructs a string reference from an `std::string` object. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2012-12-25 13:25:14 -08:00
										 |  |  |   StringRef(const std::string &s) : data_(s.c_str()), size_(s.size()) {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Converts a string reference to an `std::string` object. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2012-12-25 13:25:14 -08:00
										 |  |  |   operator std::string() const { return std::string(data_, size()); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Returns the pointer to a C string. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2012-12-25 13:25:14 -08:00
										 |  |  |   const char *c_str() const { return data_; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Returns the string size. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2012-12-25 13:25:14 -08:00
										 |  |  |   std::size_t size() const { | 
					
						
							|  |  |  |     if (size_ == 0) size_ = std::strlen(data_); | 
					
						
							|  |  |  |     return size_; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  | class FormatError : public std::runtime_error { | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2012-12-12 07:44:41 -08:00
										 |  |  |   explicit FormatError(const std::string &message) | 
					
						
							|  |  |  |   : std::runtime_error(message) {} | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-22 14:05:56 -08:00
										 |  |  | enum Alignment { | 
					
						
							|  |  |  |   ALIGN_DEFAULT, ALIGN_LEFT, ALIGN_RIGHT, ALIGN_CENTER, ALIGN_NUMERIC | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | // Flags.
 | 
					
						
							|  |  |  | enum { SIGN_FLAG = 1, PLUS_FLAG = 2, HASH_FLAG = 4 }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct Spec {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <char TYPE> | 
					
						
							|  |  |  | struct TypeSpec : Spec { | 
					
						
							|  |  |  |   Alignment align() const { return ALIGN_DEFAULT; } | 
					
						
							|  |  |  |   unsigned width() const { return 0; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool sign_flag() const { return false; } | 
					
						
							|  |  |  |   bool plus_flag() const { return false; } | 
					
						
							|  |  |  |   bool hash_flag() const { return false; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   char type() const { return TYPE; } | 
					
						
							|  |  |  |   char fill() const { return ' '; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct WidthSpec { | 
					
						
							|  |  |  |   unsigned width_; | 
					
						
							|  |  |  |   char fill_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WidthSpec(unsigned width, char fill) : width_(width), fill_(fill) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   unsigned width() const { return width_; } | 
					
						
							|  |  |  |   char fill() const { return fill_; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct AlignSpec : WidthSpec { | 
					
						
							|  |  |  |   Alignment align_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   AlignSpec(unsigned width, char fill) | 
					
						
							|  |  |  |   : WidthSpec(width, fill), align_(ALIGN_DEFAULT) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Alignment align() const { return align_; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <char TYPE> | 
					
						
							|  |  |  | struct AlignTypeSpec : AlignSpec { | 
					
						
							|  |  |  |   AlignTypeSpec(unsigned width, char fill) : AlignSpec(width, fill) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool sign_flag() const { return false; } | 
					
						
							|  |  |  |   bool plus_flag() const { return false; } | 
					
						
							|  |  |  |   bool hash_flag() const { return false; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   char type() const { return TYPE; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct FormatSpec : AlignSpec { | 
					
						
							|  |  |  |   unsigned flags_; | 
					
						
							|  |  |  |   char type_; | 
					
						
							| 
									
										
										
										
											2012-12-21 09:12:04 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |   FormatSpec(unsigned width = 0, char type = 0, char fill = ' ') | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |   : AlignSpec(width, fill), flags_(0), type_(type) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Alignment align() const { return align_; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool sign_flag() const { return (flags_ & SIGN_FLAG) != 0; } | 
					
						
							|  |  |  |   bool plus_flag() const { return (flags_ & PLUS_FLAG) != 0; } | 
					
						
							|  |  |  |   bool hash_flag() const { return (flags_ & HASH_FLAG) != 0; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   char type() const { return type_; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-12 10:17:56 -08:00
										 |  |  | template <typename T, typename SpecT> | 
					
						
							|  |  |  | class IntFormatter : public SpecT { | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |  private: | 
					
						
							|  |  |  |   T value_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2013-01-12 10:17:56 -08:00
										 |  |  |   IntFormatter(T value, const SpecT &spec = SpecT()) | 
					
						
							|  |  |  |   : SpecT(spec), value_(value) {} | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   T value() const { return value_; } | 
					
						
							| 
									
										
										
										
											2012-12-20 20:10:55 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-22 11:06:56 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |   Returns an integer formatter that formats the value in base 8. | 
					
						
							|  |  |  |   */ | 
					
						
							|  |  |  | IntFormatter<int, TypeSpec<'o'> > oct(int value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |   Returns an integer formatter that formats the value in base 16 using | 
					
						
							|  |  |  |   lower-case letters for the digits above 9. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | IntFormatter<int, TypeSpec<'x'> > hex(int value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |   Returns an integer formatter that formats the value in base 16 using | 
					
						
							|  |  |  |   upper-case letters for the digits above 9. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | IntFormatter<int, TypeSpec<'X'> > hexu(int value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |   \rst | 
					
						
							|  |  |  |   Returns an integer formatter that pads the formatted argument with the fill | 
					
						
							|  |  |  |   character to the specified width using the default (right) alignment. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   **Example**:: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  |     std::string s = str(BasicWriter() << pad(hex(0xcafe), 8, '0')); | 
					
						
							| 
									
										
										
										
											2013-01-22 11:06:56 -08:00
										 |  |  |     // s == "0000cafe"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   \endrst | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template <char TYPE_CODE> | 
					
						
							|  |  |  | IntFormatter<int, AlignTypeSpec<TYPE_CODE> > pad( | 
					
						
							|  |  |  |     int value, unsigned width, char fill = ' '); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 09:29:37 -08:00
										 |  |  | #define DEFINE_INT_FORMATTERS(TYPE) \
 | 
					
						
							|  |  |  | inline IntFormatter<TYPE, TypeSpec<'o'> > oct(TYPE value) { \ | 
					
						
							|  |  |  |   return IntFormatter<TYPE, TypeSpec<'o'> >(value, TypeSpec<'o'>()); \ | 
					
						
							|  |  |  | } \ | 
					
						
							|  |  |  |  \ | 
					
						
							|  |  |  | inline IntFormatter<TYPE, TypeSpec<'x'> > hex(TYPE value) { \ | 
					
						
							|  |  |  |   return IntFormatter<TYPE, TypeSpec<'x'> >(value, TypeSpec<'x'>()); \ | 
					
						
							|  |  |  | } \ | 
					
						
							|  |  |  |  \ | 
					
						
							|  |  |  | inline IntFormatter<TYPE, TypeSpec<'X'> > hexu(TYPE value) { \ | 
					
						
							|  |  |  |   return IntFormatter<TYPE, TypeSpec<'X'> >(value, TypeSpec<'X'>()); \ | 
					
						
							|  |  |  | } \ | 
					
						
							|  |  |  |  \ | 
					
						
							|  |  |  | template <char TYPE_CODE> \ | 
					
						
							|  |  |  | inline IntFormatter<TYPE, AlignTypeSpec<TYPE_CODE> > pad( \ | 
					
						
							|  |  |  |     IntFormatter<TYPE, TypeSpec<TYPE_CODE> > f, \ | 
					
						
							|  |  |  |     unsigned width, char fill = ' ') { \ | 
					
						
							|  |  |  |   return IntFormatter<TYPE, AlignTypeSpec<TYPE_CODE> >( \ | 
					
						
							|  |  |  |       f.value(), AlignTypeSpec<TYPE_CODE>(width, fill)); \ | 
					
						
							|  |  |  | } \ | 
					
						
							|  |  |  |  \ | 
					
						
							|  |  |  | inline IntFormatter<TYPE, AlignTypeSpec<0> > pad( \ | 
					
						
							|  |  |  |     TYPE value, unsigned width, char fill = ' ') { \ | 
					
						
							|  |  |  |   return IntFormatter<TYPE, AlignTypeSpec<0> >( \ | 
					
						
							|  |  |  |       value, AlignTypeSpec<0>(width, fill)); \ | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 09:29:37 -08:00
										 |  |  | DEFINE_INT_FORMATTERS(int) | 
					
						
							|  |  |  | DEFINE_INT_FORMATTERS(long) | 
					
						
							|  |  |  | DEFINE_INT_FORMATTERS(unsigned) | 
					
						
							|  |  |  | DEFINE_INT_FORMATTERS(unsigned long) | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | class BasicWriter { | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   static void FormatDecimal(Char *buffer, uint64_t value, unsigned num_digits); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-25 17:55:41 -08:00
										 |  |  |  protected: | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   static Char *FillPadding(Char *buffer, | 
					
						
							|  |  |  |       unsigned total_size, std::size_t content_size, char fill); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-25 17:55:41 -08:00
										 |  |  |   enum { INLINE_BUFFER_SIZE = 500 }; | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   mutable internal::Array<Char, INLINE_BUFFER_SIZE> buffer_;  // Output buffer.
 | 
					
						
							| 
									
										
										
										
											2012-12-25 17:55:41 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Grows the buffer by n characters and returns a pointer to the newly
 | 
					
						
							|  |  |  |   // allocated area.
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   Char *GrowBuffer(std::size_t n) { | 
					
						
							| 
									
										
										
										
											2012-12-25 17:55:41 -08:00
										 |  |  |     std::size_t size = buffer_.size(); | 
					
						
							|  |  |  |     buffer_.resize(size + n); | 
					
						
							|  |  |  |     return &buffer_[size]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   Char *PrepareFilledBuffer(unsigned size, const Spec &, char sign) { | 
					
						
							|  |  |  |     Char *p = GrowBuffer(size); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     *p = sign; | 
					
						
							|  |  |  |     return p + size - 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   Char *PrepareFilledBuffer(unsigned size, const AlignSpec &spec, char sign); | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Formats an integer.
 | 
					
						
							|  |  |  |   template <typename T> | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |   void FormatInt(T value, const FormatSpec &spec) { | 
					
						
							|  |  |  |     *this << IntFormatter<T, FormatSpec>(value, spec); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Formats a floating point number (double or long double).
 | 
					
						
							|  |  |  |   template <typename T> | 
					
						
							|  |  |  |   void FormatDouble(T value, const FormatSpec &spec, int precision); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   char *FormatString(const char *s, std::size_t size, const FormatSpec &spec); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-25 17:55:41 -08:00
										 |  |  |  public: | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Returns the number of characters written to the output buffer. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   std::size_t size() const { return buffer_.size(); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |     Returns a pointer to the output buffer content. No terminating null | 
					
						
							|  |  |  |     character is appended. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   const Char *data() const { return &buffer_[0]; } | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							|  |  |  |     Returns a pointer to the output buffer content with terminating null | 
					
						
							|  |  |  |     character appended. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   const Char *c_str() const { | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |     std::size_t size = buffer_.size(); | 
					
						
							|  |  |  |     buffer_.reserve(size + 1); | 
					
						
							|  |  |  |     buffer_[size] = '\0'; | 
					
						
							|  |  |  |     return &buffer_[0]; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /**
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  |     Returns the content of the output buffer as an `std::string`. | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |    */ | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   std::basic_string<Char> str() const { | 
					
						
							|  |  |  |     return std::basic_string<Char>(&buffer_[0], buffer_.size()); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  |   BasicWriter &operator<<(int value) { | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     return *this << IntFormatter<int, TypeSpec<0> >(value, TypeSpec<0>()); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  |   BasicWriter &operator<<(unsigned value) { | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     return *this << IntFormatter<unsigned, TypeSpec<0> >(value, TypeSpec<0>()); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  |   BasicWriter &operator<<(Char value) { | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |     *GrowBuffer(1) = value; | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     return *this; | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  |   BasicWriter &operator<<(const Char *value) { | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |     std::size_t size = std::strlen(value); | 
					
						
							|  |  |  |     std::strncpy(GrowBuffer(size), value, size); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     return *this; | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |   template <typename T, typename Spec> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  |   BasicWriter &operator<<(const IntFormatter<T, Spec> &f); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   void Write(const std::basic_string<char> &s, const FormatSpec &spec) { | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     FormatString(s.data(), s.size(), spec); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   void Clear() { | 
					
						
							|  |  |  |     buffer_.clear(); | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-12-25 17:55:41 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  | // Fills the padding around the content and returns the pointer to the
 | 
					
						
							|  |  |  | // content area.
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | Char *BasicWriter<Char>::FillPadding(Char *buffer, | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     unsigned total_size, std::size_t content_size, char fill) { | 
					
						
							|  |  |  |   std::size_t padding = total_size - content_size; | 
					
						
							|  |  |  |   std::size_t left_padding = padding / 2; | 
					
						
							|  |  |  |   std::fill_n(buffer, left_padding, fill); | 
					
						
							|  |  |  |   buffer += left_padding; | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   Char *content = buffer; | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   std::fill_n(buffer + content_size, padding - left_padding, fill); | 
					
						
							|  |  |  |   return content; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | void BasicWriter<Char>::FormatDecimal( | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     Char *buffer, uint64_t value, unsigned num_digits) { | 
					
						
							|  |  |  |   --num_digits; | 
					
						
							|  |  |  |   while (value >= 100) { | 
					
						
							|  |  |  |     // Integer division is slow so do it for a group of two digits instead
 | 
					
						
							|  |  |  |     // of for every digit. The idea comes from the talk by Alexandrescu
 | 
					
						
							|  |  |  |     // "Three Optimization Tips for C++". See speed-test for a comparison.
 | 
					
						
							|  |  |  |     unsigned index = (value % 100) * 2; | 
					
						
							|  |  |  |     value /= 100; | 
					
						
							|  |  |  |     buffer[num_digits] = internal::DIGITS[index + 1]; | 
					
						
							|  |  |  |     buffer[num_digits - 1] = internal::DIGITS[index]; | 
					
						
							|  |  |  |     num_digits -= 2; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (value < 10) { | 
					
						
							|  |  |  |     *buffer = static_cast<char>('0' + value); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   unsigned index = static_cast<unsigned>(value * 2); | 
					
						
							|  |  |  |   buffer[1] = internal::DIGITS[index + 1]; | 
					
						
							|  |  |  |   buffer[0] = internal::DIGITS[index]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | Char *BasicWriter<Char>::PrepareFilledBuffer( | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     unsigned size, const AlignSpec &spec, char sign) { | 
					
						
							|  |  |  |   unsigned width = spec.width(); | 
					
						
							|  |  |  |   if (width <= size) { | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     Char *p = GrowBuffer(size); | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     *p = sign; | 
					
						
							|  |  |  |     return p + size - 1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   char *p = GrowBuffer(width); | 
					
						
							|  |  |  |   char *end = p + width; | 
					
						
							|  |  |  |   Alignment align = spec.align(); | 
					
						
							|  |  |  |   if (align == ALIGN_LEFT) { | 
					
						
							|  |  |  |     *p = sign; | 
					
						
							|  |  |  |     p += size; | 
					
						
							|  |  |  |     std::fill(p, end, spec.fill()); | 
					
						
							|  |  |  |   } else if (align == ALIGN_CENTER) { | 
					
						
							|  |  |  |     p = FillPadding(p, width, size, spec.fill()); | 
					
						
							|  |  |  |     *p = sign; | 
					
						
							|  |  |  |     p += size; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (align == ALIGN_NUMERIC) { | 
					
						
							|  |  |  |       if (sign) { | 
					
						
							|  |  |  |         *p++ = sign; | 
					
						
							|  |  |  |         --size; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       *(end - size) = sign; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     std::fill(p, end - size, spec.fill()); | 
					
						
							|  |  |  |     p = end; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return p - 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							|  |  |  | template <typename T> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | void BasicWriter<Char>::FormatDouble( | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     T value, const FormatSpec &spec, int precision) { | 
					
						
							|  |  |  |   // Check type.
 | 
					
						
							|  |  |  |   char type = spec.type(); | 
					
						
							|  |  |  |   bool upper = false; | 
					
						
							|  |  |  |   switch (type) { | 
					
						
							|  |  |  |   case 0: | 
					
						
							|  |  |  |     type = 'g'; | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case 'e': case 'f': case 'g': | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   case 'F': | 
					
						
							|  |  |  | #ifdef _MSC_VER
 | 
					
						
							|  |  |  |     // MSVC's printf doesn't support 'F'.
 | 
					
						
							|  |  |  |     type = 'f'; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     // Fall through.
 | 
					
						
							|  |  |  |   case 'E': case 'G': | 
					
						
							|  |  |  |     upper = true; | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   default: | 
					
						
							|  |  |  |     internal::ReportUnknownType(type, "double"); | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   char sign = 0; | 
					
						
							|  |  |  |   // Use SignBit instead of value < 0 because the latter is always
 | 
					
						
							|  |  |  |   // false for NaN.
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   if (internal::SignBit(value)) { | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     sign = '-'; | 
					
						
							|  |  |  |     value = -value; | 
					
						
							|  |  |  |   } else if (spec.sign_flag()) { | 
					
						
							|  |  |  |     sign = spec.plus_flag() ? '+' : ' '; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (value != value) { | 
					
						
							|  |  |  |     // Format NaN ourselves because sprintf's output is not consistent
 | 
					
						
							|  |  |  |     // across platforms.
 | 
					
						
							|  |  |  |     std::size_t size = 4; | 
					
						
							|  |  |  |     const char *nan = upper ? " NAN" : " nan"; | 
					
						
							|  |  |  |     if (!sign) { | 
					
						
							|  |  |  |       --size; | 
					
						
							|  |  |  |       ++nan; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     char *out = FormatString(nan, size, spec); | 
					
						
							|  |  |  |     if (sign) | 
					
						
							|  |  |  |       *out = sign; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 08:11:04 -08:00
										 |  |  |   if (internal::IsInf(value)) { | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     // Format infinity ourselves because sprintf's output is not consistent
 | 
					
						
							|  |  |  |     // across platforms.
 | 
					
						
							|  |  |  |     std::size_t size = 4; | 
					
						
							|  |  |  |     const char *inf = upper ? " INF" : " inf"; | 
					
						
							|  |  |  |     if (!sign) { | 
					
						
							|  |  |  |       --size; | 
					
						
							|  |  |  |       ++inf; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     char *out = FormatString(inf, size, spec); | 
					
						
							|  |  |  |     if (sign) | 
					
						
							|  |  |  |       *out = sign; | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   std::size_t offset = buffer_.size(); | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   unsigned width = spec.width(); | 
					
						
							|  |  |  |   if (sign) { | 
					
						
							|  |  |  |     buffer_.reserve(buffer_.size() + std::max(width, 1u)); | 
					
						
							|  |  |  |     if (width > 0) | 
					
						
							|  |  |  |       --width; | 
					
						
							|  |  |  |     ++offset; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Build format string.
 | 
					
						
							|  |  |  |   enum { MAX_FORMAT_SIZE = 10}; // longest format: %#-*.*Lg
 | 
					
						
							|  |  |  |   char format[MAX_FORMAT_SIZE]; | 
					
						
							|  |  |  |   char *format_ptr = format; | 
					
						
							|  |  |  |   *format_ptr++ = '%'; | 
					
						
							|  |  |  |   unsigned width_for_sprintf = width; | 
					
						
							|  |  |  |   if (spec.hash_flag()) | 
					
						
							|  |  |  |     *format_ptr++ = '#'; | 
					
						
							|  |  |  |   if (spec.align() == ALIGN_CENTER) { | 
					
						
							|  |  |  |     width_for_sprintf = 0; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (spec.align() == ALIGN_LEFT) | 
					
						
							|  |  |  |       *format_ptr++ = '-'; | 
					
						
							|  |  |  |     if (width != 0) | 
					
						
							|  |  |  |       *format_ptr++ = '*'; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (precision >= 0) { | 
					
						
							|  |  |  |     *format_ptr++ = '.'; | 
					
						
							|  |  |  |     *format_ptr++ = '*'; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (internal::IsLongDouble<T>::VALUE) | 
					
						
							|  |  |  |     *format_ptr++ = 'L'; | 
					
						
							|  |  |  |   *format_ptr++ = type; | 
					
						
							|  |  |  |   *format_ptr = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Format using snprintf.
 | 
					
						
							|  |  |  |   for (;;) { | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     std::size_t size = buffer_.capacity() - offset; | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     int n = 0; | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     Char *start = &buffer_[offset]; | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     if (width_for_sprintf == 0) { | 
					
						
							|  |  |  |       n = precision < 0 ? | 
					
						
							|  |  |  |           snprintf(start, size, format, value) : | 
					
						
							|  |  |  |           snprintf(start, size, format, precision, value); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       n = precision < 0 ? | 
					
						
							|  |  |  |           snprintf(start, size, format, width_for_sprintf, value) : | 
					
						
							|  |  |  |           snprintf(start, size, format, width_for_sprintf, precision, value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (n >= 0 && offset + n < buffer_.capacity()) { | 
					
						
							|  |  |  |       if (sign) { | 
					
						
							|  |  |  |         if ((spec.align() != ALIGN_RIGHT && spec.align() != ALIGN_DEFAULT) || | 
					
						
							|  |  |  |             *start != ' ') { | 
					
						
							|  |  |  |           *(start - 1) = sign; | 
					
						
							|  |  |  |           sign = 0; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           *(start - 1) = spec.fill(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         ++n; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (spec.align() == ALIGN_CENTER && | 
					
						
							|  |  |  |           spec.width() > static_cast<unsigned>(n)) { | 
					
						
							|  |  |  |         char *p = GrowBuffer(spec.width()); | 
					
						
							|  |  |  |         std::copy(p, p + n, p + (spec.width() - n) / 2); | 
					
						
							|  |  |  |         FillPadding(p, spec.width(), n, spec.fill()); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (spec.fill() != ' ' || sign) { | 
					
						
							|  |  |  |         while (*start == ' ') | 
					
						
							|  |  |  |           *start++ = spec.fill(); | 
					
						
							|  |  |  |         if (sign) | 
					
						
							|  |  |  |           *(start - 1) = sign; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       GrowBuffer(n); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     buffer_.reserve(n >= 0 ? offset + n + 1 : 2 * buffer_.capacity()); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | char *BasicWriter<Char>::FormatString( | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     const char *s, std::size_t size, const FormatSpec &spec) { | 
					
						
							|  |  |  |   char *out = 0; | 
					
						
							|  |  |  |   if (spec.width() > size) { | 
					
						
							|  |  |  |     out = GrowBuffer(spec.width()); | 
					
						
							|  |  |  |     if (spec.align() == ALIGN_RIGHT) { | 
					
						
							|  |  |  |       std::fill_n(out, spec.width() - size, spec.fill()); | 
					
						
							|  |  |  |       out += spec.width() - size; | 
					
						
							|  |  |  |     } else if (spec.align() == ALIGN_CENTER) { | 
					
						
							|  |  |  |       out = FillPadding(out, spec.width(), size, spec.fill()); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       std::fill_n(out + size, spec.width() - size, spec.fill()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     out = GrowBuffer(size); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   std::copy(s, s + size, out); | 
					
						
							|  |  |  |   return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | template <typename T, typename Spec> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | BasicWriter<Char> &BasicWriter<Char>::operator<<( | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     const IntFormatter<T, Spec> &f) { | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |   T value = f.value(); | 
					
						
							|  |  |  |   unsigned size = 0; | 
					
						
							|  |  |  |   char sign = 0; | 
					
						
							|  |  |  |   typedef typename internal::IntTraits<T>::UnsignedType UnsignedType; | 
					
						
							|  |  |  |   UnsignedType abs_value = value; | 
					
						
							|  |  |  |   if (internal::IntTraits<T>::IsNegative(value)) { | 
					
						
							|  |  |  |     sign = '-'; | 
					
						
							|  |  |  |     ++size; | 
					
						
							|  |  |  |     abs_value = 0 - abs_value; | 
					
						
							|  |  |  |   } else if (f.sign_flag()) { | 
					
						
							|  |  |  |     sign = f.plus_flag() ? '+' : ' '; | 
					
						
							|  |  |  |     ++size; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   switch (f.type()) { | 
					
						
							|  |  |  |   case 0: case 'd': { | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     unsigned num_digits = internal::CountDigits(abs_value); | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     Char *p = PrepareFilledBuffer(size + num_digits, f, sign) - num_digits + 1; | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  |     BasicWriter::FormatDecimal(p, abs_value, num_digits); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   case 'x': case 'X': { | 
					
						
							|  |  |  |     UnsignedType n = abs_value; | 
					
						
							|  |  |  |     bool print_prefix = f.hash_flag(); | 
					
						
							|  |  |  |     if (print_prefix) size += 2; | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |       ++size; | 
					
						
							|  |  |  |     } while ((n >>= 4) != 0); | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     Char *p = PrepareFilledBuffer(size, f, sign); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     n = abs_value; | 
					
						
							|  |  |  |     const char *digits = f.type() == 'x' ? | 
					
						
							|  |  |  |         "0123456789abcdef" : "0123456789ABCDEF"; | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |       *p-- = digits[n & 0xf]; | 
					
						
							|  |  |  |     } while ((n >>= 4) != 0); | 
					
						
							|  |  |  |     if (print_prefix) { | 
					
						
							|  |  |  |       *p-- = f.type(); | 
					
						
							|  |  |  |       *p = '0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   case 'o': { | 
					
						
							|  |  |  |     UnsignedType n = abs_value; | 
					
						
							|  |  |  |     bool print_prefix = f.hash_flag(); | 
					
						
							|  |  |  |     if (print_prefix) ++size; | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |       ++size; | 
					
						
							|  |  |  |     } while ((n >>= 3) != 0); | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     Char *p = PrepareFilledBuffer(size, f, sign); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     n = abs_value; | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |       *p-- = '0' + (n & 7); | 
					
						
							|  |  |  |     } while ((n >>= 3) != 0); | 
					
						
							|  |  |  |     if (print_prefix) | 
					
						
							|  |  |  |       *p = '0'; | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   default: | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |     internal::ReportUnknownType(f.type(), "integer"); | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return *this; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | typedef BasicWriter<char> Writer; | 
					
						
							|  |  |  | typedef BasicWriter<wchar_t> WWriter; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  | // The default formatting function.
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  | template <typename Char, typename T> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | void Format(BasicWriter<Char> &f, const FormatSpec &spec, const T &value) { | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   std::basic_ostringstream<Char> os; | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  |   os << value; | 
					
						
							|  |  |  |   f.Write(os.str(), spec); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-01-08 09:56:05 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 08:05:01 -08:00
										 |  |  | namespace internal { | 
					
						
							|  |  |  | // Formats an argument of a custom type, such as a user-defined class.
 | 
					
						
							|  |  |  | template <typename Char, typename T> | 
					
						
							|  |  |  | void FormatCustomArg( | 
					
						
							|  |  |  |     BasicWriter<Char> &w, const void *arg, const FormatSpec &spec) { | 
					
						
							|  |  |  |   Format(w, spec, *static_cast<const T*>(arg)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-29 09:27:26 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |   \rst | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   The :cpp:class:`fmt::BasicFormatter` template provides string formatting | 
					
						
							| 
									
										
										
										
											2012-12-29 09:27:26 -08:00
										 |  |  |   functionality similar to Python's `str.format | 
					
						
							|  |  |  |   <http://docs.python.org/3/library/stdtypes.html#str.format>`__.
 | 
					
						
							|  |  |  |   The output is stored in a memory buffer that grows dynamically. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-03 08:57:34 -08:00
										 |  |  |   **Example**:: | 
					
						
							| 
									
										
										
										
											2012-12-29 09:27:26 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |      Formatter out; | 
					
						
							|  |  |  |      out("Current point:\n"); | 
					
						
							|  |  |  |      out("(-{:+f}, {:+f})") << 3.14 << -3.14; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   This will populate the buffer of the ``out`` object with the following | 
					
						
							|  |  |  |   output: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   .. code-block:: none | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |      Current point: | 
					
						
							|  |  |  |      (-3.140000, +3.140000) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   The buffer can be accessed using :meth:`data` or :meth:`c_str`. | 
					
						
							|  |  |  |   \endrst | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | template <typename Char> | 
					
						
							|  |  |  | class BasicFormatter : public BasicWriter<Char> { | 
					
						
							| 
									
										
										
										
											2012-12-29 09:27:26 -08:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |   enum Type { | 
					
						
							| 
									
										
										
										
											2012-12-09 11:32:39 -08:00
										 |  |  |     // Numeric types should go first.
 | 
					
						
							| 
									
										
										
										
											2012-12-09 14:13:23 -08:00
										 |  |  |     INT, UINT, LONG, ULONG, DOUBLE, LONG_DOUBLE, | 
					
						
							|  |  |  |     LAST_NUMERIC_TYPE = LONG_DOUBLE, | 
					
						
							|  |  |  |     CHAR, STRING, WSTRING, POINTER, CUSTOM | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 08:05:01 -08:00
										 |  |  |   typedef void (*FormatFunc)( | 
					
						
							|  |  |  |       BasicWriter<Char> &w, const void *arg, const FormatSpec &spec); | 
					
						
							| 
									
										
										
										
											2012-12-09 11:32:39 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |   // A format argument.
 | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  |   class Arg { | 
					
						
							|  |  |  |    private: | 
					
						
							|  |  |  |     // This method is private to disallow formatting of arbitrary pointers.
 | 
					
						
							|  |  |  |     // If you want to output a pointer cast it to const void*. Do not implement!
 | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     Arg(const T *value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // This method is private to disallow formatting of arbitrary pointers.
 | 
					
						
							|  |  |  |     // If you want to output a pointer cast it to void*. Do not implement!
 | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							|  |  |  |     Arg(T *value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // This method is private to disallow formatting of wide characters.
 | 
					
						
							|  |  |  |     // If you want to output a wide character cast it to integer type.
 | 
					
						
							|  |  |  |     // Do not implement!
 | 
					
						
							|  |  |  |     Arg(wchar_t value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    public: | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |     Type type; | 
					
						
							|  |  |  |     union { | 
					
						
							|  |  |  |       int int_value; | 
					
						
							|  |  |  |       unsigned uint_value; | 
					
						
							|  |  |  |       double double_value; | 
					
						
							| 
									
										
										
										
											2013-01-13 09:29:37 -08:00
										 |  |  |       long long_value; | 
					
						
							|  |  |  |       unsigned long ulong_value; | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |       long double long_double_value; | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |       const void *pointer_value; | 
					
						
							| 
									
										
										
										
											2012-12-08 08:17:12 -08:00
										 |  |  |       struct { | 
					
						
							| 
									
										
										
										
											2012-12-11 21:47:05 -08:00
										 |  |  |         const char *value; | 
					
						
							| 
									
										
										
										
											2012-12-09 09:03:47 -08:00
										 |  |  |         std::size_t size; | 
					
						
							| 
									
										
										
										
											2012-12-11 21:47:05 -08:00
										 |  |  |       } string; | 
					
						
							| 
									
										
										
										
											2012-12-09 09:03:47 -08:00
										 |  |  |       struct { | 
					
						
							| 
									
										
										
										
											2012-12-11 21:47:05 -08:00
										 |  |  |         const void *value; | 
					
						
							| 
									
										
										
										
											2012-12-09 11:32:39 -08:00
										 |  |  |         FormatFunc format; | 
					
						
							| 
									
										
										
										
											2012-12-11 21:47:05 -08:00
										 |  |  |       } custom; | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     mutable BasicFormatter *formatter; | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 21:47:05 -08:00
										 |  |  |     Arg(int value) : type(INT), int_value(value), formatter(0) {} | 
					
						
							|  |  |  |     Arg(unsigned value) : type(UINT), uint_value(value), formatter(0) {} | 
					
						
							|  |  |  |     Arg(long value) : type(LONG), long_value(value), formatter(0) {} | 
					
						
							|  |  |  |     Arg(unsigned long value) : type(ULONG), ulong_value(value), formatter(0) {} | 
					
						
							|  |  |  |     Arg(double value) : type(DOUBLE), double_value(value), formatter(0) {} | 
					
						
							|  |  |  |     Arg(long double value) | 
					
						
							|  |  |  |     : type(LONG_DOUBLE), long_double_value(value), formatter(0) {} | 
					
						
							|  |  |  |     Arg(char value) : type(CHAR), int_value(value), formatter(0) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Arg(const char *value) : type(STRING), formatter(0) { | 
					
						
							|  |  |  |       string.value = value; | 
					
						
							|  |  |  |       string.size = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Arg(char *value) : type(STRING), formatter(0) { | 
					
						
							|  |  |  |       string.value = value; | 
					
						
							|  |  |  |       string.size = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Arg(const void *value) | 
					
						
							|  |  |  |     : type(POINTER), pointer_value(value), formatter(0) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Arg(void *value) : type(POINTER), pointer_value(value), formatter(0) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Arg(const std::string &value) : type(STRING), formatter(0) { | 
					
						
							|  |  |  |       string.value = value.c_str(); | 
					
						
							|  |  |  |       string.size = value.size(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     template <typename T> | 
					
						
							| 
									
										
										
										
											2012-12-11 21:47:05 -08:00
										 |  |  |     Arg(const T &value) : type(CUSTOM), formatter(0) { | 
					
						
							|  |  |  |       custom.value = &value; | 
					
						
							| 
									
										
										
										
											2013-02-05 08:05:01 -08:00
										 |  |  |       custom.format = &internal::FormatCustomArg<Char, T>; | 
					
						
							| 
									
										
										
										
											2012-12-11 21:47:05 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ~Arg() { | 
					
						
							| 
									
										
										
										
											2012-12-11 13:54:53 -08:00
										 |  |  |       // Format is called here to make sure that a referred object is
 | 
					
						
							|  |  |  |       // still alive, for example:
 | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  |       //
 | 
					
						
							|  |  |  |       //   Print("{0}") << std::string("test");
 | 
					
						
							|  |  |  |       //
 | 
					
						
							|  |  |  |       // Here an Arg object refers to a temporary std::string which is
 | 
					
						
							|  |  |  |       // destroyed at the end of the statement. Since the string object is
 | 
					
						
							|  |  |  |       // constructed before the Arg object, it will be destroyed after,
 | 
					
						
							| 
									
										
										
										
											2012-12-11 13:54:53 -08:00
										 |  |  |       // so it will be alive in the Arg's destructor where Format is called.
 | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  |       // Note that the string object will not necessarily be alive when
 | 
					
						
							| 
									
										
										
										
											2012-12-11 16:26:04 -08:00
										 |  |  |       // the destructor of ArgInserter is called.
 | 
					
						
							| 
									
										
										
										
											2012-12-17 14:56:44 -08:00
										 |  |  |       formatter->CompleteFormatting(); | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 20:37:35 -08:00
										 |  |  |   enum { NUM_INLINE_ARGS = 10 }; | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |   internal::Array<const Arg*, NUM_INLINE_ARGS> args_;  // Format arguments.
 | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   const Char *format_;  // Format string.
 | 
					
						
							| 
									
										
										
										
											2012-12-12 15:21:11 -08:00
										 |  |  |   int num_open_braces_; | 
					
						
							| 
									
										
										
										
											2012-12-27 06:56:55 -08:00
										 |  |  |   int next_arg_index_; | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   friend class internal::ArgInserter<Char>; | 
					
						
							|  |  |  |   friend class internal::FormatterProxy<Char>; | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   void Add(const Arg &arg) { | 
					
						
							| 
									
										
										
										
											2012-12-11 10:27:13 -08:00
										 |  |  |     args_.push_back(&arg); | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-25 13:25:14 -08:00
										 |  |  |   void ReportError(const char *s, StringRef message) const; | 
					
						
							| 
									
										
										
										
											2012-12-12 15:21:11 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   unsigned ParseUInt(const char *&s) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Parses argument index and returns an argument with this index.
 | 
					
						
							| 
									
										
										
										
											2012-12-27 06:56:55 -08:00
										 |  |  |   const Arg &ParseArgIndex(const char *&s); | 
					
						
							| 
									
										
										
										
											2012-12-12 15:21:11 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-25 13:45:12 -08:00
										 |  |  |   void CheckSign(const char *&s, const Arg &arg); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  |   void DoFormat(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-17 14:56:44 -08:00
										 |  |  |   void CompleteFormatting() { | 
					
						
							| 
									
										
										
										
											2012-12-11 12:23:52 -08:00
										 |  |  |     if (!format_) return; | 
					
						
							|  |  |  |     DoFormat(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2013-01-04 07:04:35 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Constructs a formatter with an empty output buffer. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   BasicFormatter() : format_(0) {} | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-04 07:04:35 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Formats a string appending the output to the internal buffer. | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  |     Arguments are accepted through the returned `ArgInserter` object | 
					
						
							|  |  |  |     using inserter operator `<<`. | 
					
						
							| 
									
										
										
										
											2013-01-04 07:04:35 -08:00
										 |  |  |   */ | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   internal::ArgInserter<Char> operator()(StringRef format); | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | typedef BasicFormatter<char> Formatter; | 
					
						
							|  |  |  | typedef BasicFormatter<wchar_t> WFormatter; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | inline std::basic_string<Char> str(const BasicWriter<Char> &f) { | 
					
						
							| 
									
										
										
										
											2013-01-26 16:07:28 -08:00
										 |  |  |   return f.str(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-02-02 20:29:02 -08:00
										 |  |  | inline const Char *c_str(const BasicWriter<Char> &f) { return f.c_str(); } | 
					
						
							| 
									
										
										
										
											2013-01-22 11:06:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | std::string str(internal::FormatterProxy<char> p); | 
					
						
							|  |  |  | const char *c_str(internal::FormatterProxy<char> p); | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | namespace internal { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-21 11:42:25 -08:00
										 |  |  | using fmt::str; | 
					
						
							|  |  |  | using fmt::c_str; | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2013-01-13 09:47:01 -08:00
										 |  |  | class FormatterProxy { | 
					
						
							|  |  |  |  private: | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   BasicFormatter<Char> *formatter_; | 
					
						
							| 
									
										
										
										
											2013-01-13 09:47:01 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   explicit FormatterProxy(BasicFormatter<Char> *f) : formatter_(f) {} | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   BasicFormatter<Char> *Format() { | 
					
						
							| 
									
										
										
										
											2013-01-13 09:47:01 -08:00
										 |  |  |     formatter_->CompleteFormatting(); | 
					
						
							|  |  |  |     return formatter_; | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | // This is a transient object that normally exists only as a temporary
 | 
					
						
							|  |  |  | // returned by one of the formatting functions. It stores a reference
 | 
					
						
							|  |  |  | // to a formatter and provides operator<< that feeds arguments to the
 | 
					
						
							|  |  |  | // formatter.
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | template <typename Char> | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | class ArgInserter { | 
					
						
							|  |  |  |  private: | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   mutable BasicFormatter<Char> *formatter_; | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   friend class fmt::BasicFormatter<Char>; | 
					
						
							| 
									
										
										
										
											2013-01-21 11:42:25 -08:00
										 |  |  |   friend class fmt::StringRef; | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-14 19:01:48 -08:00
										 |  |  |   // Do not implement.
 | 
					
						
							|  |  |  |   void operator=(const ArgInserter& other); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |  protected: | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   explicit ArgInserter(BasicFormatter<Char> *f = 0) : formatter_(f) {} | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   void Init(BasicFormatter<Char> &f, const char *format) { | 
					
						
							| 
									
										
										
										
											2012-12-14 19:01:48 -08:00
										 |  |  |     const ArgInserter &other = f(format); | 
					
						
							|  |  |  |     formatter_ = other.formatter_; | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |     other.formatter_ = 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-14 19:01:48 -08:00
										 |  |  |   ArgInserter(const ArgInserter& other) | 
					
						
							|  |  |  |   : formatter_(other.formatter_) { | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |     other.formatter_ = 0; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   const BasicFormatter<Char> *Format() const { | 
					
						
							|  |  |  |     BasicFormatter<Char> *f = formatter_; | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |     if (f) { | 
					
						
							|  |  |  |       formatter_ = 0; | 
					
						
							| 
									
										
										
										
											2012-12-17 14:56:44 -08:00
										 |  |  |       f->CompleteFormatting(); | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     return f; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   BasicFormatter<Char> *formatter() const { return formatter_; } | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |   const char *format() const { return formatter_->format_; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-14 19:01:48 -08:00
										 |  |  |   void ResetFormatter() const { formatter_ = 0; } | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							|  |  |  |   ~ArgInserter() { | 
					
						
							|  |  |  |     if (formatter_) | 
					
						
							| 
									
										
										
										
											2012-12-17 14:56:44 -08:00
										 |  |  |       formatter_->CompleteFormatting(); | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Feeds an argument to a formatter.
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   ArgInserter &operator<<(const typename BasicFormatter<Char>::Arg &arg) { | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  |     arg.formatter = formatter_; | 
					
						
							|  |  |  |     formatter_->Add(arg); | 
					
						
							|  |  |  |     return *this; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   operator FormatterProxy<Char>() { | 
					
						
							|  |  |  |     BasicFormatter<Char> *f = formatter_; | 
					
						
							| 
									
										
										
										
											2012-12-16 10:03:19 -08:00
										 |  |  |     formatter_ = 0; | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     return FormatterProxy<Char>(f); | 
					
						
							| 
									
										
										
										
											2012-12-16 10:03:19 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-19 10:47:00 -08:00
										 |  |  |   operator StringRef() { | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     const BasicFormatter<Char> *f = Format(); | 
					
						
							| 
									
										
										
										
											2012-12-19 10:47:00 -08:00
										 |  |  |     return StringRef(f->c_str(), f->size()); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-12-12 09:14:00 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |   Returns the content of the output buffer as an `std::string`. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | inline std::string str(internal::FormatterProxy<char> p) { | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  |   return p.Format()->str(); | 
					
						
							| 
									
										
										
										
											2012-12-17 14:56:44 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |   Returns a pointer to the output buffer content with terminating null | 
					
						
							|  |  |  |   character appended. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | inline const char *c_str(internal::FormatterProxy<char> p) { | 
					
						
							| 
									
										
										
										
											2013-01-13 07:50:28 -08:00
										 |  |  |   return p.Format()->c_str(); | 
					
						
							| 
									
										
										
										
											2012-12-09 09:03:47 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | template <typename Char> | 
					
						
							|  |  |  | inline internal::ArgInserter<Char> | 
					
						
							|  |  |  | BasicFormatter<Char>::operator()(StringRef format) { | 
					
						
							|  |  |  |   internal::ArgInserter<Char> inserter(this); | 
					
						
							| 
									
										
										
										
											2012-12-29 09:27:26 -08:00
										 |  |  |   format_ = format.c_str(); | 
					
						
							| 
									
										
										
										
											2012-12-10 17:16:08 -08:00
										 |  |  |   args_.clear(); | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   return inserter; | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-12 10:08:39 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |   A formatting action that does nothing. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-01-13 07:14:54 -08:00
										 |  |  | class NoAction { | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2013-01-12 10:08:39 -08:00
										 |  |  |   /** Does nothing. */ | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   template <typename Char> | 
					
						
							|  |  |  |   void operator()(const BasicFormatter<Char> &) const {} | 
					
						
							| 
									
										
										
										
											2013-01-03 08:57:34 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-04 09:14:34 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |   A formatter with an action performed when formatting is complete. | 
					
						
							|  |  |  |   Objects of this class normally exist only as temporaries returned | 
					
						
							|  |  |  |   by one of the formatting functions which explains the name. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | template <typename Char, typename Action = NoAction> | 
					
						
							|  |  |  | class TempFormatter : public internal::ArgInserter<Char> { | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   BasicFormatter<Char> formatter_; | 
					
						
							| 
									
										
										
										
											2012-12-11 16:26:04 -08:00
										 |  |  |   Action action_; | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-15 20:17:03 -08:00
										 |  |  |   // Forbid copying other than from a temporary. Do not implement.
 | 
					
						
							| 
									
										
										
										
											2012-12-16 20:57:35 -08:00
										 |  |  |   TempFormatter(TempFormatter &); | 
					
						
							| 
									
										
										
										
											2012-12-15 20:17:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |   // Do not implement.
 | 
					
						
							| 
									
										
										
										
											2012-12-16 20:57:35 -08:00
										 |  |  |   TempFormatter& operator=(const TempFormatter &); | 
					
						
							| 
									
										
										
										
											2012-12-15 20:17:03 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   struct Proxy { | 
					
						
							|  |  |  |     const char *format; | 
					
						
							|  |  |  |     Action action; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Proxy(const char *fmt, Action a) : format(fmt), action(a) {} | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2013-01-12 10:08:39 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     \rst | 
					
						
							|  |  |  |     Constructs a temporary formatter with a format string and an action. | 
					
						
							|  |  |  |     The action should be an unary function object that takes a const | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     reference to :cpp:class:`fmt::BasicFormatter` as an argument. | 
					
						
							| 
									
										
										
										
											2013-01-21 11:42:25 -08:00
										 |  |  |     See :cpp:class:`fmt::NoAction` and :cpp:class:`fmt::Write` for | 
					
						
							| 
									
										
										
										
											2013-01-12 10:08:39 -08:00
										 |  |  |     examples of action classes. | 
					
						
							|  |  |  |     \endrst | 
					
						
							|  |  |  |   */ | 
					
						
							| 
									
										
										
										
											2013-01-03 08:57:34 -08:00
										 |  |  |   explicit TempFormatter(StringRef format, Action a = Action()) | 
					
						
							| 
									
										
										
										
											2012-12-11 16:26:04 -08:00
										 |  |  |   : action_(a) { | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     this->Init(formatter_, format.c_str()); | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-12 10:08:39 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Constructs a temporary formatter from a proxy object. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2012-12-16 20:57:35 -08:00
										 |  |  |   TempFormatter(const Proxy &p) | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   : internal::ArgInserter<Char>(0), action_(p.action) { | 
					
						
							|  |  |  |     this->Init(formatter_, p.format); | 
					
						
							| 
									
										
										
										
											2012-12-11 20:49:01 -08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-12-08 08:17:12 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-12 10:08:39 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Performs the actual formatting, invokes the action and destroys the object. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2012-12-16 20:57:35 -08:00
										 |  |  |   ~TempFormatter() { | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     if (this->formatter()) | 
					
						
							|  |  |  |       action_(*this->Format()); | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-12-15 20:17:03 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-12 10:08:39 -08:00
										 |  |  |   /**
 | 
					
						
							|  |  |  |     Converts a temporary formatter into a proxy object. | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2012-12-15 20:17:03 -08:00
										 |  |  |   operator Proxy() { | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |     const char *fmt = this->format(); | 
					
						
							|  |  |  |     this->ResetFormatter(); | 
					
						
							| 
									
										
										
										
											2012-12-15 20:17:03 -08:00
										 |  |  |     return Proxy(fmt, action_); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-03 08:57:34 -08:00
										 |  |  | /**
 | 
					
						
							|  |  |  |   \rst | 
					
						
							|  |  |  |   Formats a string. Returns a temporary formatter object that accepts | 
					
						
							|  |  |  |   arguments via operator ``<<``. *format* is a format string that contains | 
					
						
							|  |  |  |   literal text and replacement fields surrounded by braces ``{}``. | 
					
						
							|  |  |  |   The formatter object replaces the fields with formatted arguments | 
					
						
							|  |  |  |   and stores the output in a memory buffer. The content of the buffer can | 
					
						
							| 
									
										
										
										
											2013-01-21 11:42:25 -08:00
										 |  |  |   be converted to ``std::string`` with :cpp:func:`fmt::str()` or | 
					
						
							|  |  |  |   accessed as a C string with :cpp:func:`fmt::c_str()`. | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-03 08:57:34 -08:00
										 |  |  |   **Example**:: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-12 10:08:39 -08:00
										 |  |  |     std::string message = str(Format("The answer is {}") << 42); | 
					
						
							| 
									
										
										
										
											2013-01-03 08:57:34 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   See also `Format String Syntax`_. | 
					
						
							|  |  |  |   \endrst | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | inline TempFormatter<char> Format(StringRef format) { | 
					
						
							|  |  |  |   return TempFormatter<char>(format); | 
					
						
							| 
									
										
										
										
											2012-12-11 13:54:53 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 16:26:04 -08:00
										 |  |  | // A formatting action that writes formatted output to stdout.
 | 
					
						
							| 
									
										
										
										
											2012-12-11 13:54:53 -08:00
										 |  |  | struct Write { | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |   void operator()(const BasicFormatter<char> &f) const { | 
					
						
							| 
									
										
										
										
											2012-12-11 13:54:53 -08:00
										 |  |  |     std::fwrite(f.data(), 1, f.size(), stdout); | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2012-12-11 13:54:53 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-11 16:26:04 -08:00
										 |  |  | // Formats a string and prints it to stdout.
 | 
					
						
							|  |  |  | // Example:
 | 
					
						
							|  |  |  | //   Print("Elapsed time: {0:.2f} seconds") << 1.23;
 | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  | inline TempFormatter<char, Write> Print(StringRef format) { | 
					
						
							|  |  |  |   return TempFormatter<char, Write>(format); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Throws Exception(message) if format contains '}', otherwise throws
 | 
					
						
							|  |  |  | // FormatError reporting unmatched '{'. The idea is that unmatched '{'
 | 
					
						
							|  |  |  | // should override other errors.
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							|  |  |  | void BasicFormatter<Char>::ReportError(const char *s, StringRef message) const { | 
					
						
							|  |  |  |   for (int num_open_braces = num_open_braces_; *s; ++s) { | 
					
						
							|  |  |  |     if (*s == '{') { | 
					
						
							|  |  |  |       ++num_open_braces; | 
					
						
							|  |  |  |     } else if (*s == '}') { | 
					
						
							|  |  |  |       if (--num_open_braces == 0) | 
					
						
							|  |  |  |         throw fmt::FormatError(message); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   throw fmt::FormatError("unmatched '{' in format"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Parses an unsigned integer advancing s to the end of the parsed input.
 | 
					
						
							|  |  |  | // This function assumes that the first character of s is a digit.
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							|  |  |  | unsigned BasicFormatter<Char>::ParseUInt(const char *&s) const { | 
					
						
							|  |  |  |   assert('0' <= *s && *s <= '9'); | 
					
						
							|  |  |  |   unsigned value = 0; | 
					
						
							|  |  |  |   do { | 
					
						
							|  |  |  |     unsigned new_value = value * 10 + (*s++ - '0'); | 
					
						
							|  |  |  |     if (new_value < value)  // Check if value wrapped around.
 | 
					
						
							|  |  |  |       ReportError(s, "number is too big in format"); | 
					
						
							|  |  |  |     value = new_value; | 
					
						
							|  |  |  |   } while ('0' <= *s && *s <= '9'); | 
					
						
							|  |  |  |   return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							|  |  |  | inline const typename BasicFormatter<Char>::Arg | 
					
						
							|  |  |  |     &BasicFormatter<Char>::ParseArgIndex(const char *&s) { | 
					
						
							|  |  |  |   unsigned arg_index = 0; | 
					
						
							|  |  |  |   if (*s < '0' || *s > '9') { | 
					
						
							|  |  |  |     if (*s != '}' && *s != ':') | 
					
						
							|  |  |  |       ReportError(s, "invalid argument index in format string"); | 
					
						
							|  |  |  |     if (next_arg_index_ < 0) { | 
					
						
							|  |  |  |       ReportError(s, | 
					
						
							|  |  |  |           "cannot switch from manual to automatic argument indexing"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     arg_index = next_arg_index_++; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (next_arg_index_ > 0) { | 
					
						
							|  |  |  |       ReportError(s, | 
					
						
							|  |  |  |           "cannot switch from automatic to manual argument indexing"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     next_arg_index_ = -1; | 
					
						
							|  |  |  |     arg_index = ParseUInt(s); | 
					
						
							|  |  |  |     if (arg_index >= args_.size()) | 
					
						
							|  |  |  |       ReportError(s, "argument index is out of range in format"); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return *args_[arg_index]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							|  |  |  | void BasicFormatter<Char>::CheckSign(const char *&s, const Arg &arg) { | 
					
						
							|  |  |  |   if (arg.type > LAST_NUMERIC_TYPE) { | 
					
						
							|  |  |  |     ReportError(s, | 
					
						
							|  |  |  |         Format("format specifier '{0}' requires numeric argument") << *s); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   if (arg.type == UINT || arg.type == ULONG) { | 
					
						
							|  |  |  |     ReportError(s, | 
					
						
							|  |  |  |         Format("format specifier '{0}' requires signed argument") << *s); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   ++s; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Char> | 
					
						
							|  |  |  | void BasicFormatter<Char>::DoFormat() { | 
					
						
							|  |  |  |   const Char *start = format_; | 
					
						
							|  |  |  |   format_ = 0; | 
					
						
							|  |  |  |   next_arg_index_ = 0; | 
					
						
							|  |  |  |   const Char *s = start; | 
					
						
							|  |  |  |   while (*s) { | 
					
						
							|  |  |  |     char c = *s++; | 
					
						
							|  |  |  |     if (c != '{' && c != '}') continue; | 
					
						
							|  |  |  |     if (*s == c) { | 
					
						
							|  |  |  |       this->buffer_.append(start, s); | 
					
						
							|  |  |  |       start = ++s; | 
					
						
							|  |  |  |       continue; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (c == '}') | 
					
						
							|  |  |  |       throw FormatError("unmatched '}' in format"); | 
					
						
							|  |  |  |     num_open_braces_= 1; | 
					
						
							|  |  |  |     this->buffer_.append(start, s - 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Arg &arg = ParseArgIndex(s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FormatSpec spec; | 
					
						
							|  |  |  |     int precision = -1; | 
					
						
							|  |  |  |     if (*s == ':') { | 
					
						
							|  |  |  |       ++s; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Parse fill and alignment.
 | 
					
						
							|  |  |  |       if (char c = *s) { | 
					
						
							|  |  |  |         const char *p = s + 1; | 
					
						
							|  |  |  |         spec.align_ = ALIGN_DEFAULT; | 
					
						
							|  |  |  |         do { | 
					
						
							|  |  |  |           switch (*p) { | 
					
						
							|  |  |  |           case '<': | 
					
						
							|  |  |  |             spec.align_ = ALIGN_LEFT; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case '>': | 
					
						
							|  |  |  |             spec.align_ = ALIGN_RIGHT; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case '=': | 
					
						
							|  |  |  |             spec.align_ = ALIGN_NUMERIC; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case '^': | 
					
						
							|  |  |  |             spec.align_ = ALIGN_CENTER; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           if (spec.align_ != ALIGN_DEFAULT) { | 
					
						
							|  |  |  |             if (p != s) { | 
					
						
							|  |  |  |               if (c == '}') break; | 
					
						
							|  |  |  |               if (c == '{') | 
					
						
							|  |  |  |                 ReportError(s, "invalid fill character '{'"); | 
					
						
							|  |  |  |               s += 2; | 
					
						
							|  |  |  |               spec.fill_ = c; | 
					
						
							|  |  |  |             } else ++s; | 
					
						
							|  |  |  |             if (spec.align_ == ALIGN_NUMERIC && arg.type > LAST_NUMERIC_TYPE) | 
					
						
							|  |  |  |               ReportError(s, "format specifier '=' requires numeric argument"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } while (--p >= s); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Parse sign.
 | 
					
						
							|  |  |  |       switch (*s) { | 
					
						
							|  |  |  |       case '+': | 
					
						
							|  |  |  |         CheckSign(s, arg); | 
					
						
							|  |  |  |         spec.flags_ |= SIGN_FLAG | PLUS_FLAG; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case '-': | 
					
						
							|  |  |  |         CheckSign(s, arg); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case ' ': | 
					
						
							|  |  |  |         CheckSign(s, arg); | 
					
						
							|  |  |  |         spec.flags_ |= SIGN_FLAG; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (*s == '#') { | 
					
						
							|  |  |  |         if (arg.type > LAST_NUMERIC_TYPE) | 
					
						
							|  |  |  |           ReportError(s, "format specifier '#' requires numeric argument"); | 
					
						
							|  |  |  |         spec.flags_ |= HASH_FLAG; | 
					
						
							|  |  |  |         ++s; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Parse width and zero flag.
 | 
					
						
							|  |  |  |       if ('0' <= *s && *s <= '9') { | 
					
						
							|  |  |  |         if (*s == '0') { | 
					
						
							|  |  |  |           if (arg.type > LAST_NUMERIC_TYPE) | 
					
						
							|  |  |  |             ReportError(s, "format specifier '0' requires numeric argument"); | 
					
						
							|  |  |  |           spec.align_ = ALIGN_NUMERIC; | 
					
						
							|  |  |  |           spec.fill_ = '0'; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         // Zero may be parsed again as a part of the width, but it is simpler
 | 
					
						
							|  |  |  |         // and more efficient than checking if the next char is a digit.
 | 
					
						
							|  |  |  |         unsigned value = ParseUInt(s); | 
					
						
							|  |  |  |         if (value > INT_MAX) | 
					
						
							|  |  |  |           ReportError(s, "number is too big in format"); | 
					
						
							|  |  |  |         spec.width_ = value; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Parse precision.
 | 
					
						
							|  |  |  |       if (*s == '.') { | 
					
						
							|  |  |  |         ++s; | 
					
						
							|  |  |  |         precision = 0; | 
					
						
							|  |  |  |         if ('0' <= *s && *s <= '9') { | 
					
						
							|  |  |  |           unsigned value = ParseUInt(s); | 
					
						
							|  |  |  |           if (value > INT_MAX) | 
					
						
							|  |  |  |             ReportError(s, "number is too big in format"); | 
					
						
							|  |  |  |           precision = value; | 
					
						
							|  |  |  |         } else if (*s == '{') { | 
					
						
							|  |  |  |           ++s; | 
					
						
							|  |  |  |           ++num_open_braces_; | 
					
						
							|  |  |  |           const Arg &precision_arg = ParseArgIndex(s); | 
					
						
							|  |  |  |           unsigned long value = 0; | 
					
						
							|  |  |  |           switch (precision_arg.type) { | 
					
						
							|  |  |  |           case INT: | 
					
						
							|  |  |  |             if (precision_arg.int_value < 0) | 
					
						
							|  |  |  |               ReportError(s, "negative precision in format"); | 
					
						
							|  |  |  |             value = precision_arg.int_value; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case UINT: | 
					
						
							|  |  |  |             value = precision_arg.uint_value; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case LONG: | 
					
						
							|  |  |  |             if (precision_arg.long_value < 0) | 
					
						
							|  |  |  |               ReportError(s, "negative precision in format"); | 
					
						
							|  |  |  |             value = precision_arg.long_value; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           case ULONG: | 
					
						
							|  |  |  |             value = precision_arg.ulong_value; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |           default: | 
					
						
							|  |  |  |             ReportError(s, "precision is not integer"); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           if (value > INT_MAX) | 
					
						
							|  |  |  |             ReportError(s, "number is too big in format"); | 
					
						
							|  |  |  |           precision = value; | 
					
						
							|  |  |  |           if (*s++ != '}') | 
					
						
							|  |  |  |             throw FormatError("unmatched '{' in format"); | 
					
						
							|  |  |  |           --num_open_braces_; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           ReportError(s, "missing precision in format"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (arg.type != DOUBLE && arg.type != LONG_DOUBLE) { | 
					
						
							|  |  |  |           ReportError(s, | 
					
						
							|  |  |  |               "precision specifier requires floating-point argument"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Parse type.
 | 
					
						
							|  |  |  |       if (*s != '}' && *s) | 
					
						
							|  |  |  |         spec.type_ = *s++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (*s++ != '}') | 
					
						
							|  |  |  |       throw FormatError("unmatched '{' in format"); | 
					
						
							|  |  |  |     start = s; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Format argument.
 | 
					
						
							|  |  |  |     switch (arg.type) { | 
					
						
							|  |  |  |     case INT: | 
					
						
							|  |  |  |       this->FormatInt(arg.int_value, spec); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case UINT: | 
					
						
							|  |  |  |       this->FormatInt(arg.uint_value, spec); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case LONG: | 
					
						
							|  |  |  |       this->FormatInt(arg.long_value, spec); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case ULONG: | 
					
						
							|  |  |  |       this->FormatInt(arg.ulong_value, spec); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case DOUBLE: | 
					
						
							|  |  |  |       this->FormatDouble(arg.double_value, spec, precision); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case LONG_DOUBLE: | 
					
						
							|  |  |  |       this->FormatDouble(arg.long_double_value, spec, precision); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case CHAR: { | 
					
						
							|  |  |  |       if (spec.type_ && spec.type_ != 'c') | 
					
						
							|  |  |  |         internal::ReportUnknownType(spec.type_, "char"); | 
					
						
							|  |  |  |       char *out = 0; | 
					
						
							|  |  |  |       if (spec.width_ > 1) { | 
					
						
							|  |  |  |         out = this->GrowBuffer(spec.width_); | 
					
						
							|  |  |  |         if (spec.align_ == ALIGN_RIGHT) { | 
					
						
							|  |  |  |           std::fill_n(out, spec.width_ - 1, spec.fill_); | 
					
						
							|  |  |  |           out += spec.width_ - 1; | 
					
						
							|  |  |  |         } else if (spec.align_ == ALIGN_CENTER) { | 
					
						
							|  |  |  |           out = this->FillPadding(out, spec.width_, 1, spec.fill_); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           std::fill_n(out + 1, spec.width_ - 1, spec.fill_); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         out = this->GrowBuffer(1); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       *out = arg.int_value; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case STRING: { | 
					
						
							|  |  |  |       if (spec.type_ && spec.type_ != 's') | 
					
						
							|  |  |  |         internal::ReportUnknownType(spec.type_, "string"); | 
					
						
							|  |  |  |       const char *str = arg.string.value; | 
					
						
							|  |  |  |       std::size_t size = arg.string.size; | 
					
						
							|  |  |  |       if (size == 0) { | 
					
						
							|  |  |  |         if (!str) | 
					
						
							|  |  |  |           throw FormatError("string pointer is null"); | 
					
						
							|  |  |  |         if (*str) | 
					
						
							|  |  |  |           size = std::strlen(str); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       this->FormatString(str, size, spec); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case POINTER: | 
					
						
							|  |  |  |       if (spec.type_ && spec.type_ != 'p') | 
					
						
							|  |  |  |         internal::ReportUnknownType(spec.type_, "pointer"); | 
					
						
							|  |  |  |       spec.flags_= HASH_FLAG; | 
					
						
							|  |  |  |       spec.type_ = 'x'; | 
					
						
							|  |  |  |       this->FormatInt(reinterpret_cast<uintptr_t>(arg.pointer_value), spec); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     case CUSTOM: | 
					
						
							|  |  |  |       if (spec.type_) | 
					
						
							|  |  |  |         internal::ReportUnknownType(spec.type_, "object"); | 
					
						
							| 
									
										
										
										
											2013-02-05 08:05:01 -08:00
										 |  |  |       arg.custom.format(*this, arg.custom.value, spec); | 
					
						
							| 
									
										
										
										
											2013-02-05 07:28:54 -08:00
										 |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       assert(false); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   this->buffer_.append(start, s); | 
					
						
							| 
									
										
										
										
											2012-12-11 13:54:53 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-12-07 08:31:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif  // FORMAT_H_
 |