Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/pch/src/util/Convert.h @ 3181

Last change on this file since 3181 was 3152, checked in by rgrieder, 16 years ago

Removed ConvertValue function, non capital version should be enough ;)
Also added std::string forward declaration to OrxonoxConfig.h.in

  • Property svn:eol-style set to native
File size: 14.9 KB
Line 
1/*
2 *   ORXONOX - the hottest 3D action shooter ever to exist
3 *                    > www.orxonox.net <
4 *
5 *
6 *   License notice:
7 *
8 *   This program is free software; you can redistribute it and/or
9 *   modify it under the terms of the GNU General Public License
10 *   as published by the Free Software Foundation; either version 2
11 *   of the License, or (at your option) any later version.
12 *
13 *   This program is distributed in the hope that it will be useful,
14 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
15 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 *   GNU General Public License for more details.
17 *
18 *   You should have received a copy of the GNU General Public License
19 *   along with this program; if not, write to the Free Software
20 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
21 *
22 *   Author:
23 *      Reto Grieder
24 *      Fabian 'x3n' Landau
25 *      Benjamin Grauer
26 *   Co-authors:
27 *      ...
28 */
29
30/*!
31    @file
32    @brief Definition and Implementation of the Convert class.
33*/
34
35#ifndef _Converter_H__
36#define _Converter_H__
37
38#include "UtilPrereqs.h"
39
40#include <string>
41#include <sstream>
42#include <typeinfo>
43
44#include "Debug.h"
45#include "String.h"
46
47// GCC generates warnings when implicitely casting from float to int for instance.
48// This is however exactly what convertValue does, so we need to suppress these warnings.
49// They only occur when using the ImplicitConversion template.
50#ifdef ORXONOX_COMPILER_GCC
51#  pragma GCC system_header
52#endif
53
54
55///////////////////////////////////////////////
56// Static detection for conversion functions //
57///////////////////////////////////////////////
58
59/* The idea to use the sizeof() operator on return functions to determine function existance
60   is described in 'Modern C++ design' by Alexandrescu (2001). */
61
62// disable warnings about possible loss of data
63#ifdef ORXONOX_COMPILER_MSVC
64#  pragma warning(push)
65#  pragma warning(disable:4244)
66#endif
67
68namespace orxonox
69{
70    template <class FromType, class ToType>
71    class ImplicitConversion
72    {
73    private:
74        ImplicitConversion(); ImplicitConversion(const ImplicitConversion&); ~ImplicitConversion();
75        // Gets chosen only if there is an implicit conversion from FromType to ToType.
76        static char test(ToType);
77        // Accepts any argument. Why do we not use a template? The reason is that with templates,
78        // the function above is only taken iff it is an exact type match. But since we want to
79        // check for implicit conversion, we have to use the ellipsis.
80        static long long test(...);
81        static FromType object; // helper object to handle private c'tor and d'tor
82    public:
83        // test(object) only has 'long long' return type iff the compiler doesn't choose test(...)
84        enum { exists = (sizeof(test(object)) == sizeof(char)) };
85    };
86}
87
88#ifdef ORXONOX_COMPILER_MSVC
89#  pragma warning(pop)
90#endif
91
92
93////////////////////////////////////
94//// ACTUAL CONVERSION SEQUENCE ////
95////////////////////////////////////
96/*
97    There is a distinct priority when choosing the right conversion function:
98    Overwrite:
99    1. (Partial) template specialisation of ConverterExplicit::convert()
100    Fallbacks:
101    2. Any possible implicit conversion. This includes 'FooBar' --> 'int' if FooBar defines operator float().
102    3. Global or member operators for stringstream when converting from or to std::string (or FROM const char*)
103    4. (Partial) template specialisation of ConverterFallback::convert()
104    5. Function that simply displays "Could not convert value" with type information obtained from typeid().
105
106    Notes:
107    There has to be an exact type match when using template specialisations.
108    Template specialisations can be defined after including this file. Any implicit cast function or iostream
109    operator has to be declared BEFORE this file gets parsed.
110
111    Defining your own functions:
112    There are obviously 4 ways to specifiy a user defined conversion. What should I use?
113
114    Usually, ConverterFallback fits quite well. You won't have to deal with the conversion from
115    'MyClass' to 'MyClass' by using another explicit template specialisation to avoid ambiguities.
116
117    However if you want to overwrite an implicit conversion or an iostream operator, you really need to
118    make use of ConverterExplicit.
119*/
120
121namespace orxonox
122{
123    namespace
124    {
125        //! Little template that maps integers to entire types (Alexandrescu 2001)
126        template <int I>
127        struct Int2Type { };
128    }
129
130
131    ///////////////////
132    // No Conversion //
133    ///////////////////
134
135    // Default template. No conversion available at all.
136    template <class FromType, class ToType>
137    struct ConverterFallback
138    {
139        FORCEINLINE static bool convert(ToType* output, const FromType& input)
140        {
141            COUT(2) << "Could not convert value of type " << typeid(FromType).name()
142                    << " to type " << typeid(ToType).name() << std::endl;
143            return false;
144        }
145    };
146
147    // If all else fails, try a dynamic_cast for pointer types.
148    template <class FromType, class ToType>
149    struct ConverterFallback<FromType*, ToType*>
150    {
151        FORCEINLINE static bool convert(ToType** output, FromType* const input)
152        {
153            ToType* temp = dynamic_cast<ToType*>(input);
154            if (temp)
155            {
156                *output = temp;
157                return true;
158            }
159            else
160                return false;
161        }
162    };
163}
164
165
166///////////////////////
167// ConverterFallback //
168///////////////////////
169
170// Default template for stringstream
171template <class FromType, class ToType>
172struct ConverterStringStream
173{
174    FORCEINLINE static bool convert(ToType* output, const FromType& input)
175    {
176        return orxonox::ConverterFallback<FromType, ToType>::convert(output, input);
177    }
178};
179
180
181/////////////
182// OStream //
183/////////////
184
185namespace fallbackTemplates
186{
187    template <class FromType>
188    FORCEINLINE bool operator <<(std::ostream& outstream,  const FromType& input)
189    {
190        std::string temp;
191        if (orxonox::ConverterFallback<FromType, std::string>::convert(&temp, input))
192        {
193            std::operator <<(outstream, temp);
194            return true;
195        }
196        else
197            return false;
198    }
199}
200
201// template that evaluates whether we can convert to std::string via ostringstream
202template <class FromType>
203struct ConverterStringStream<FromType, std::string>
204{
205    FORCEINLINE static bool convert(std::string* output, const FromType& input)
206    {
207        using namespace fallbackTemplates;
208        // this operator call only chooses fallbackTemplates::operator<< if there's no other fitting function
209        std::ostringstream oss;
210        if (oss << input)
211        {
212            (*output) = oss.str();
213            return true;
214        }
215        else
216            return false;
217    }
218};
219
220
221/////////////
222// IStream //
223/////////////
224
225namespace fallbackTemplates
226{
227    template <class ToType>
228    FORCEINLINE bool operator >>(std::istream& instream, ToType& output)
229    {
230        return orxonox::ConverterFallback<std::string, ToType>
231            ::convert(&output, static_cast<std::istringstream&>(instream).str());
232    }
233}
234
235// template that evaluates whether we can convert from std::string via ostringstream
236template <class ToType>
237struct ConverterStringStream<std::string, ToType>
238{
239    FORCEINLINE static bool convert(ToType* output, const std::string& input)
240    {
241        using namespace fallbackTemplates;
242        std::istringstream iss(input);
243        // this operator call only chooses fallbackTemplates::operator>> if there's no other fitting function
244        if (iss >> (*output))
245        {
246            return true;
247        }
248        else
249            return false;
250    }
251};
252
253namespace orxonox
254{
255
256    ///////////////////
257    // Implicit Cast //
258    ///////////////////
259
260    // implicit cast not possible, try stringstream conversion next
261    template <class FromType, class ToType>
262    FORCEINLINE bool convertImplicitely(ToType* output, const FromType& input, orxonox::Int2Type<false>)
263    {
264        return ConverterStringStream<FromType, ToType>::convert(output, input);
265    }
266
267    // We can cast implicitely
268    template <class FromType, class ToType>
269    FORCEINLINE bool convertImplicitely(ToType* output, const FromType& input, orxonox::Int2Type<true>)
270    {
271        (*output) = static_cast<ToType>(input);
272        return true;
273    }
274
275
276    ////////////////////////////////
277    // ConverterExplicit Fallback //
278    ////////////////////////////////
279
280    // Default template if no specialisation is available
281    template <class FromType, class ToType>
282    struct ConverterExplicit
283    {
284        FORCEINLINE static bool convert(ToType* output, const FromType& input)
285        {
286            // Try implict cast and probe first. If a simple cast is not possible, it will not compile
287            // We therefore have to out source it into another template function
288            const bool probe = ImplicitConversion<FromType, ToType>::exists;
289            return convertImplicitely(output, input, orxonox::Int2Type<probe>());
290        }
291    };
292
293
294    //////////////////////
295    // Public Functions //
296    //////////////////////
297
298    /**
299    @brief
300        Converts any value to any other as long as there exists a conversion.
301        Otherwise, the conversion will generate a runtime warning and return false.
302        For information about the different conversion methods (user defined too), see the section
303        'Actual conversion sequence' in this file above.
304    */
305    template <class FromType, class ToType>
306    FORCEINLINE bool convertValue(ToType* output, const FromType& input)
307    {
308        return ConverterExplicit<FromType, ToType>::convert(output, input);
309    }
310
311    // For compatibility reasons. The same, but with capital ConvertValue
312    template<class FromType, class ToType>
313    FORCEINLINE bool ConvertValue(ToType* output, const FromType& input)
314    {
315        return convertValue(output, input);
316    }
317
318    // Calls convertValue and returns true if the conversion was successful.
319    // Otherwise the fallback is used.
320    /**
321    @brief
322        Converts any value to any other as long as there exists a conversion.
323        Otherwise, the conversion will generate a runtime warning and return false.
324        For information about the different conversion methods (user defined too), see the section
325        'Actual conversion sequence' in this file above.
326        If the conversion doesn't succeed, 'fallback' is written to '*output'.
327    @param fallback
328        A default value that gets written to '*output' if there is no conversion.
329    */
330    template<class FromType, class ToType>
331    FORCEINLINE bool convertValue(ToType* output, const FromType& input, const ToType& fallback)
332    {
333        if (convertValue(output, input))
334            return true;
335        else
336        {
337            (*output) = fallback;
338            return false;
339        }
340    }
341
342    // Directly returns the converted value, even if the conversion was not successful.
343    template<class FromType, class ToType>
344    FORCEINLINE ToType getConvertedValue(const FromType& input)
345    {
346        ToType output;
347        convertValue(&output, input);
348        return output;
349    }
350
351    // Directly returns the converted value, but uses the fallback on failure.
352    template<class FromType, class ToType>
353    FORCEINLINE ToType getConvertedValue(const FromType& input, const ToType& fallback)
354    {
355        ToType output;
356        convertValue(&output, input, fallback);
357        return output;
358    }
359
360    // Like getConvertedValue, but the template argument order is in reverse.
361    // That means you can call it exactly like static_cast<ToType>(fromTypeValue).
362    template<class ToType, class FromType>
363    FORCEINLINE ToType multi_cast(const FromType& input)
364    {
365        ToType output;
366        convertValue(&output, input);
367        return output;
368    }
369
370    // convert to string Shortcut
371    template <class FromType>
372    FORCEINLINE std::string convertToString(FromType value)
373    {
374      return getConvertedValue<FromType, std::string>(value);
375    }
376
377    // convert from string Shortcut
378    template <class ToType>
379    FORCEINLINE ToType convertFromString(std::string str)
380    {
381      return getConvertedValue<std::string, ToType>(str);
382    }
383
384    ////////////////////////////////
385    // Special string conversions //
386    ////////////////////////////////
387
388    // Delegate conversion from const char* to std::string
389    template <class ToType>
390    struct ConverterExplicit<const char*, ToType>
391    {
392        FORCEINLINE static bool convert(ToType* output, const char* input)
393        {
394            return convertValue<std::string, ToType>(output, input);
395        }
396    };
397
398    // These conversions would exhibit ambiguous << or >> operators when using stringstream
399    template <>
400    struct ConverterExplicit<char, std::string>
401    {
402        FORCEINLINE static bool convert(std::string* output, const char input)
403        {
404            *output = std::string(1, input);
405            return true;
406        }
407    };
408    template <>
409    struct ConverterExplicit<unsigned char, std::string>
410    {
411        FORCEINLINE static bool convert(std::string* output, const unsigned char input)
412        {
413            *output = std::string(1, input);
414            return true;
415        }
416    };
417    template <>
418    struct ConverterExplicit<std::string, char>
419    {
420        FORCEINLINE static bool convert(char* output, const std::string input)
421        {
422            if (input != "")
423                *output = input[0];
424            else
425                *output = '\0';
426            return true;
427        }
428    };
429    template <>
430    struct ConverterExplicit<std::string, unsigned char>
431    {
432        FORCEINLINE static bool convert(unsigned char* output, const std::string input)
433        {
434            if (input != "")
435                *output = input[0];
436            else
437                *output = '\0';
438            return true;
439        }
440    };
441
442
443    // bool to std::string
444    template <>
445    struct ConverterExplicit<bool, std::string>
446    {
447        FORCEINLINE static bool convert(std::string* output, const bool& input)
448        {
449            if (input)
450              *output = "true";
451            else
452              *output = "false";
453            return true;
454        }
455    };
456
457    // std::string to bool
458    template <>
459    struct ConverterExplicit<std::string, bool>
460    {
461        static bool convert(bool* output, const std::string& input)
462        {
463            std::string stripped = getLowercase(removeTrailingWhitespaces(input));
464            if (stripped == "true" || stripped == "on" || stripped == "yes")
465            {
466              *output = true;
467              return true;
468            }
469            else if (stripped == "false" || stripped == "off" || stripped == "no")
470            {
471              *output = false;
472              return true;
473            }
474
475            std::istringstream iss(input);
476            if (iss >> (*output))
477                return true;
478            else
479                return false;
480        }
481    };
482}
483
484#endif /* _Convert_H__ */
Note: See TracBrowser for help on using the repository browser.