|
1 /* |
|
2 Copyright 2012-2014 Santeri Piippo |
|
3 All rights reserved. |
|
4 |
|
5 Redistribution and use in source and binary forms, with or without |
|
6 modification, are permitted provided that the following conditions |
|
7 are met: |
|
8 |
|
9 1. Redistributions of source code must retain the above copyright |
|
10 notice, this list of conditions and the following disclaimer. |
|
11 2. Redistributions in binary form must reproduce the above copyright |
|
12 notice, this list of conditions and the following disclaimer in the |
|
13 documentation and/or other materials provided with the distribution. |
|
14 3. The name of the author may not be used to endorse or promote products |
|
15 derived from this software without specific prior written permission. |
|
16 |
|
17 THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR |
|
18 IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES |
|
19 OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. |
|
20 IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, |
|
21 INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT |
|
22 NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
|
23 DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
|
24 THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
|
25 (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
|
26 THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
27 */ |
|
28 |
|
29 #ifndef BOTC_FORMAT_H |
|
30 #define BOTC_FORMAT_H |
|
31 |
|
32 #include "string.h" |
|
33 #include "list.h" |
|
34 |
|
35 class FormatArgument |
|
36 { |
|
37 public: |
|
38 FormatArgument (const String& a) : m_text (a) {} |
|
39 FormatArgument (char a) : m_text (a) {} |
|
40 FormatArgument (int a) : m_text (String::fromNumber (a)) {} |
|
41 FormatArgument (long a) : m_text (String::fromNumber (a)) {} |
|
42 FormatArgument (const char* a) : m_text (a) {} |
|
43 |
|
44 FormatArgument (void* a) |
|
45 { |
|
46 m_text.sprintf ("%p", a); |
|
47 } |
|
48 |
|
49 FormatArgument (const void* a) |
|
50 { |
|
51 m_text.sprintf ("%p", a); |
|
52 } |
|
53 |
|
54 template<class T> FormatArgument (const List<T>& list) |
|
55 { |
|
56 if (list.isEmpty()) |
|
57 { |
|
58 m_text = "{}"; |
|
59 return; |
|
60 } |
|
61 |
|
62 m_text = "{ "; |
|
63 |
|
64 for (const T& a : list) |
|
65 { |
|
66 if (&a != &list[0]) |
|
67 m_text += ", "; |
|
68 |
|
69 m_text += FormatArgument (a).text(); |
|
70 } |
|
71 |
|
72 m_text += " }"; |
|
73 } |
|
74 |
|
75 inline const String& text() const |
|
76 { |
|
77 return m_text; |
|
78 } |
|
79 |
|
80 private: |
|
81 String m_text; |
|
82 }; |
|
83 |
|
84 #ifndef IN_IDE_PARSER |
|
85 # ifdef DEBUG |
|
86 # define devf(...) PrintTo (stderr, __VA_ARGS__) |
|
87 # define dvalof( A ) PrintTo (stderr, "value of '%1' = %2\n", #A, A) |
|
88 # else |
|
89 # define devf(...) |
|
90 # define dvalof( A ) |
|
91 # endif // DEBUG |
|
92 #else |
|
93 // print something in debug builds |
|
94 void devf (void, ...); |
|
95 |
|
96 // print the value of @a |
|
97 void dvalof (void a); |
|
98 #endif // IN_IDE_PARSER |
|
99 |
|
100 |
|
101 /** |
|
102 * Formats the given string with the given args. |
|
103 * |
|
104 * @param fmtstr Formatter string to process. |
|
105 * @param args Args to format with the string. |
|
106 * @see format() |
|
107 */ |
|
108 String formatArgs (const String& fmtstr, const std::vector<String>& args); |
|
109 |
|
110 /** |
|
111 * Expands the given arguments into a vector of strings. |
|
112 * |
|
113 * @param data Where to insert the strings. |
|
114 * @param arg First argument to process |
|
115 * @param rest... Rest of the arguments. |
|
116 */ |
|
117 template<typename T, typename... RestTypes> |
|
118 void expandFormatArguments (std::vector<String>& data, const T& arg, const RestTypes& ... rest) |
|
119 { |
|
120 data.push_back (FormatArgument (arg).text()); |
|
121 expandFormatArguments (data, rest...); |
|
122 } |
|
123 |
|
124 /** |
|
125 * This is an overload of @c ExpandFormatArguments for end-of-args support. |
|
126 */ |
|
127 static void expandFormatArguments (std::vector<String>& data) __attribute__ ( (unused)); |
|
128 static void expandFormatArguments (std::vector<String>& data) |
|
129 { |
|
130 (void) data; |
|
131 } |
|
132 |
|
133 /** |
|
134 * Formats the given formatter string and args and returns the string. |
|
135 * This is essentially a modernized sprintf. |
|
136 * |
|
137 * Args in the format string take the form %n where n is a digit. The argument |
|
138 * will be expanded to the nth argument passed. This is essentially Qt's |
|
139 * QString::arg() syntax. Note: %0 is invalid. |
|
140 * |
|
141 * Arguments can be passed a modifier which takes the form of a character |
|
142 * just before the digit. Currently supported modifiers are s, d and x. |
|
143 * |
|
144 * - s: The argument will expand into "s" if it would've not expanded into "1" |
|
145 * otherwise. If it would have expanded into "1" it will expand into an |
|
146 * empty string. |
|
147 * |
|
148 * - d: The argument expands into the numeric form of the first character of |
|
149 * its previous expansion. Use this to get numeric forms of @c char |
|
150 * arguments. |
|
151 * |
|
152 * - x: The numeric argument will be represented in hexadecimal notation. This |
|
153 * will work if the argument is a string representing a number. If the |
|
154 * argument did not expand into a number in the first place, 0 is used |
|
155 * (and 0x0 is printed). |
|
156 * |
|
157 * @param fmtstr Formatter string to process |
|
158 * @param raw_args Arguments for the formatter string. |
|
159 * @return the formatted string. |
|
160 * @see Print |
|
161 * @see PrintTo |
|
162 */ |
|
163 template<typename... argtypes> |
|
164 String format (const String& fmtstr, const argtypes&... raw_args) |
|
165 { |
|
166 std::vector<String> args; |
|
167 expandFormatArguments (args, raw_args...); |
|
168 assert (args.size() == sizeof... (raw_args)); |
|
169 return formatArgs (fmtstr, args); |
|
170 } |
|
171 |
|
172 /** |
|
173 * This is an overload of @c format where no arguments are supplied. |
|
174 * @return the formatter string as-is. |
|
175 */ |
|
176 static String format (const String& fmtstr) __attribute__ ( (unused)); |
|
177 static String format (const String& fmtstr) |
|
178 { |
|
179 return fmtstr; |
|
180 } |
|
181 |
|
182 /** |
|
183 * Processes the given formatter string using @c format and prints it to the |
|
184 * specified file pointer. |
|
185 * |
|
186 * @param fp File pointer to print the formatted string to |
|
187 * @param fmtstr Formatter string for @c format |
|
188 * @param args Arguments for @c fmtstr |
|
189 */ |
|
190 template<typename... argtypes> |
|
191 void printTo (FILE* fp, const String& fmtstr, const argtypes&... args) |
|
192 { |
|
193 fprintf (fp, "%s", format (fmtstr, args...).c_str()); |
|
194 } |
|
195 |
|
196 /** |
|
197 * Processes the given formatter string using @c format and prints the result to |
|
198 * @c stdout. |
|
199 * |
|
200 * @param fmtstr Formatter string for @c format |
|
201 * @param args Arguments for @c fmtstr |
|
202 */ |
|
203 template<typename... argtypes> |
|
204 void print (const String& fmtstr, const argtypes&... args) |
|
205 { |
|
206 printTo (stdout, fmtstr, args...); |
|
207 } |
|
208 |
|
209 /** |
|
210 * Throws an std::runtime_error with the processed formatted string. The program |
|
211 * execution terminates after a call to this function as the exception is first |
|
212 * caught in @c main which prints the error to stderr and then exits. |
|
213 * |
|
214 * @param fmtstr The formatter string of the error. |
|
215 * @param args The args to the formatter string. |
|
216 * @see Format |
|
217 */ |
|
218 template<typename... argtypes> |
|
219 void error (const String& fmtstr, const argtypes&... args) |
|
220 { |
|
221 error (format (fmtstr, args...)); |
|
222 } |
|
223 |
|
224 /** |
|
225 * An overload of @c Error with no string formatting in between. |
|
226 * |
|
227 * @param msg The error message. |
|
228 */ |
|
229 void error (String msg); |
|
230 |
|
231 #endif // BOTC_FORMAT_H |