Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/libraries/util/Convert.h @ 7324

Last change on this file since 7324 was 7305, checked in by rgrieder, 15 years ago

File rename.

  • Property svn:eol-style set to native
File size: 12.8 KB
RevLine 
[1052]1/*
2 *   ORXONOX - the hottest 3D action shooter ever to exist
[1505]3 *                    > www.orxonox.net <
[1052]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:
[2087]23 *      Reto Grieder
[1791]24 *      Fabian 'x3n' Landau
[1052]25 *      Benjamin Grauer
[1505]26 *   Co-authors:
[1052]27 *      ...
28 */
29
30/*!
[2087]31    @file
[1052]32    @brief Definition and Implementation of the Convert class.
33*/
34
[2087]35#ifndef _Converter_H__
36#define _Converter_H__
[1052]37
[1062]38#include "UtilPrereqs.h"
39
[1052]40#include <string>
41#include <sstream>
[1837]42#include <typeinfo>
[7266]43#include <loki/TypeManip.h>
[1052]44
[1747]45#include "Debug.h"
[7305]46#include "ImplicitConversion.h"
[1064]47
[2087]48////////////////////////////////////
49//// ACTUAL CONVERSION SEQUENCE ////
50////////////////////////////////////
51/*
52    There is a distinct priority when choosing the right conversion function:
53    Overwrite:
54    1. (Partial) template specialisation of ConverterExplicit::convert()
55    Fallbacks:
56    2. Any possible implicit conversion. This includes 'FooBar' --> 'int' if FooBar defines operator float().
57    3. Global or member operators for stringstream when converting from or to std::string (or FROM const char*)
58    4. (Partial) template specialisation of ConverterFallback::convert()
59    5. Function that simply displays "Could not convert value" with type information obtained from typeid().
[1505]60
[2087]61    Notes:
62    There has to be an exact type match when using template specialisations.
63    Template specialisations can be defined after including this file. Any implicit cast function or iostream
64    operator has to be declared BEFORE this file gets parsed.
[1505]65
[2087]66    Defining your own functions:
[7163]67    There are obviously 4 ways to specify a user defined conversion. What should you use?
[1505]68
[2087]69    Usually, ConverterFallback fits quite well. You won't have to deal with the conversion from
70    'MyClass' to 'MyClass' by using another explicit template specialisation to avoid ambiguities.
[1505]71
[2087]72    However if you want to overwrite an implicit conversion or an iostream operator, you really need to
[7163]73    make use of ConverterExplicit. We have to do this for the Ogre classes for instance because they
74    define stream operators we don't particulary like.
[2087]75*/
76
[2171]77namespace orxonox
[1505]78{
[2171]79    ///////////////////
80    // No Conversion //
81    ///////////////////
[1505]82
[2171]83    // Default template. No conversion available at all.
84    template <class FromType, class ToType>
85    struct ConverterFallback
[1505]86    {
[3196]87        FORCEINLINE static bool convert(ToType* output, const FromType& input)
[2171]88        {
89            COUT(2) << "Could not convert value of type " << typeid(FromType).name()
90                    << " to type " << typeid(ToType).name() << std::endl;
91            return false;
92        }
93    };
[2087]94
[2171]95    // If all else fails, try a dynamic_cast for pointer types.
96    template <class FromType, class ToType>
97    struct ConverterFallback<FromType*, ToType*>
[1505]98    {
[3196]99        FORCEINLINE static bool convert(ToType** output, FromType* const input)
[2087]100        {
[2171]101            ToType* temp = dynamic_cast<ToType*>(input);
102            if (temp)
103            {
104                *output = temp;
105                return true;
106            }
107            else
108                return false;
[2087]109        }
[2171]110    };
[7284]111
112    ////////////
113    // upcast //
114    ////////////
115    namespace detail
116    {
117        // perform a static cast if ToType is a base of FromType
118        template<class ToType, class FromType>
119        FORCEINLINE ToType upcast(FromType input, Loki::Int2Type<true>)
120        {
121            return static_cast<ToType>(input);
122        }
123
124        // return zero if ToType is not a base of FromType
125        template<class ToType, class FromType>
126        FORCEINLINE ToType upcast(FromType input, Loki::Int2Type<false>)
127        {
128            return 0;
129        }
130    }
131
132    // performs an upcast if ToType is a base of FromType, returns zero otherwise
133    template <class ToType, class FromType>
134    FORCEINLINE ToType upcast(FromType input)
135    {
136        enum { probe = ImplicitConversion<FromType, ToType>::exists };
137        return detail::upcast<ToType, FromType>(input, Loki::Int2Type<probe>());
138    }
[2171]139}
[1505]140
141
[2087]142///////////////////////
143// ConverterFallback //
144///////////////////////
145
146// Default template for stringstream
147template <class FromType, class ToType>
148struct ConverterStringStream
[1505]149{
[3196]150    FORCEINLINE static bool convert(ToType* output, const FromType& input)
[1505]151    {
[2171]152        return orxonox::ConverterFallback<FromType, ToType>::convert(output, input);
[1505]153    }
154};
155
156
157/////////////
[2087]158// OStream //
[1505]159/////////////
160
[2087]161namespace fallbackTemplates
[1505]162{
[2087]163    template <class FromType>
[3196]164    FORCEINLINE bool operator <<(std::ostream& outstream,  const FromType& input)
[2087]165    {
166        std::string temp;
[2171]167        if (orxonox::ConverterFallback<FromType, std::string>::convert(&temp, input))
[2087]168        {
169            std::operator <<(outstream, temp);
170            return true;
171        }
172        else
173            return false;
174    }
175}
[1505]176
[2087]177// template that evaluates whether we can convert to std::string via ostringstream
[1505]178template <class FromType>
[2087]179struct ConverterStringStream<FromType, std::string>
[1505]180{
[3196]181    FORCEINLINE static bool convert(std::string* output, const FromType& input)
[1505]182    {
[2087]183        using namespace fallbackTemplates;
184        // this operator call only chooses fallbackTemplates::operator<< if there's no other fitting function
[1505]185        std::ostringstream oss;
186        if (oss << input)
187        {
188            (*output) = oss.str();
189            return true;
190        }
191        else
192            return false;
193    }
194};
195
[2087]196
197/////////////
198// IStream //
199/////////////
200
201namespace fallbackTemplates
[1625]202{
[2087]203    template <class ToType>
[3196]204    FORCEINLINE bool operator >>(std::istream& instream, ToType& output)
[2087]205    {
[2171]206        return orxonox::ConverterFallback<std::string, ToType>
[2087]207            ::convert(&output, static_cast<std::istringstream&>(instream).str());
208    }
[1625]209}
210
[2087]211// template that evaluates whether we can convert from std::string via ostringstream
[1505]212template <class ToType>
[2087]213struct ConverterStringStream<std::string, ToType>
[1505]214{
[3196]215    FORCEINLINE static bool convert(ToType* output, const std::string& input)
[1505]216    {
[2087]217        using namespace fallbackTemplates;
[1505]218        std::istringstream iss(input);
[2087]219        // this operator call only chooses fallbackTemplates::operator>> if there's no other fitting function
[1505]220        if (iss >> (*output))
[2087]221        {
[1505]222            return true;
[2087]223        }
[1505]224        else
225            return false;
226    }
227};
228
[2171]229namespace orxonox
[1625]230{
[1505]231
[2171]232    ///////////////////
233    // Implicit Cast //
234    ///////////////////
[1625]235
[2171]236    // implicit cast not possible, try stringstream conversion next
237    template <class FromType, class ToType>
[7266]238    FORCEINLINE bool convertImplicitely(ToType* output, const FromType& input, Loki::Int2Type<false>)
[1505]239    {
[2171]240        return ConverterStringStream<FromType, ToType>::convert(output, input);
[1505]241    }
242
[2171]243    // We can cast implicitely
244    template <class FromType, class ToType>
[7266]245    FORCEINLINE bool convertImplicitely(ToType* output, const FromType& input, Loki::Int2Type<true>)
[2171]246    {
247        (*output) = static_cast<ToType>(input);
[2087]248        return true;
[1505]249    }
250
251
[2171]252    ////////////////////////////////
253    // ConverterExplicit Fallback //
254    ////////////////////////////////
[1505]255
[2171]256    // Default template if no specialisation is available
257    template <class FromType, class ToType>
258    struct ConverterExplicit
259    {
[3234]260        enum { probe = ImplicitConversion<FromType, ToType>::exists };
[3196]261        FORCEINLINE static bool convert(ToType* output, const FromType& input)
[2171]262        {
263            // Try implict cast and probe first. If a simple cast is not possible, it will not compile
264            // We therefore have to out source it into another template function
[7266]265            return convertImplicitely(output, input, Loki::Int2Type<probe>());
[2171]266        }
267    };
[2087]268
269
[2171]270    //////////////////////
271    // Public Functions //
272    //////////////////////
[2087]273
[2171]274    /**
275    @brief
276        Converts any value to any other as long as there exists a conversion.
277        Otherwise, the conversion will generate a runtime warning and return false.
278        For information about the different conversion methods (user defined too), see the section
279        'Actual conversion sequence' in this file above.
280    */
281    template <class FromType, class ToType>
[3196]282    FORCEINLINE bool convertValue(ToType* output, const FromType& input)
[2171]283    {
284        return ConverterExplicit<FromType, ToType>::convert(output, input);
285    }
[2087]286
[2171]287    // Calls convertValue and returns true if the conversion was successful.
288    // Otherwise the fallback is used.
289    /**
290    @brief
291        Converts any value to any other as long as there exists a conversion.
292        Otherwise, the conversion will generate a runtime warning and return false.
293        For information about the different conversion methods (user defined too), see the section
294        'Actual conversion sequence' in this file above.
295        If the conversion doesn't succeed, 'fallback' is written to '*output'.
296    @param fallback
297        A default value that gets written to '*output' if there is no conversion.
298    */
299    template<class FromType, class ToType>
[3196]300    FORCEINLINE bool convertValue(ToType* output, const FromType& input, const ToType& fallback)
[1625]301    {
[2171]302        if (convertValue(output, input))
303            return true;
304        else
305        {
306            (*output) = fallback;
307            return false;
308        }
[1625]309    }
310
[2171]311    // Directly returns the converted value, even if the conversion was not successful.
312    template<class FromType, class ToType>
[3196]313    FORCEINLINE ToType getConvertedValue(const FromType& input)
[1505]314    {
[2171]315        ToType output;
316        convertValue(&output, input);
317        return output;
[1505]318    }
[2171]319
320    // Directly returns the converted value, but uses the fallback on failure.
321    template<class FromType, class ToType>
[3196]322    FORCEINLINE ToType getConvertedValue(const FromType& input, const ToType& fallback)
[1505]323    {
[2171]324        ToType output;
325        convertValue(&output, input, fallback);
326        return output;
[1505]327    }
[2171]328
329    // Like getConvertedValue, but the template argument order is in reverse.
330    // That means you can call it exactly like static_cast<ToType>(fromTypeValue).
331    template<class ToType, class FromType>
[3196]332    FORCEINLINE ToType multi_cast(const FromType& input)
[1505]333    {
[2171]334        ToType output;
335        convertValue(&output, input);
336        return output;
[1505]337    }
338
[2171]339    ////////////////////////////////
340    // Special string conversions //
341    ////////////////////////////////
342
343    // Delegate conversion from const char* to std::string
344    template <class ToType>
345    struct ConverterExplicit<const char*, ToType>
[1625]346    {
[3196]347        FORCEINLINE static bool convert(ToType* output, const char* input)
[1625]348        {
[2171]349            return convertValue<std::string, ToType>(output, input);
[1625]350        }
[2171]351    };
352
353    // These conversions would exhibit ambiguous << or >> operators when using stringstream
354    template <>
355    struct ConverterExplicit<char, std::string>
356    {
[3196]357        FORCEINLINE static bool convert(std::string* output, const char input)
[1625]358        {
[6417]359            *output = input;
[2171]360            return true;
[1625]361        }
[2171]362    };
363    template <>
364    struct ConverterExplicit<unsigned char, std::string>
365    {
[3196]366        FORCEINLINE static bool convert(std::string* output, const unsigned char input)
[2171]367        {
[6417]368            *output = input;
[2171]369            return true;
370        }
371    };
372    template <>
373    struct ConverterExplicit<std::string, char>
374    {
[6417]375        FORCEINLINE static bool convert(char* output, const std::string& input)
[2171]376        {
[6417]377            if (!input.empty())
[2171]378                *output = input[0];
379            else
380                *output = '\0';
381            return true;
382        }
383    };
384    template <>
385    struct ConverterExplicit<std::string, unsigned char>
386    {
[6417]387        FORCEINLINE static bool convert(unsigned char* output, const std::string& input)
[2171]388        {
[6417]389            if (!input.empty())
[2171]390                *output = input[0];
391            else
392                *output = '\0';
393            return true;
394        }
395    };
[1625]396
[2171]397
398    // bool to std::string
399    template <>
400    struct ConverterExplicit<bool, std::string>
401    {
[3196]402        FORCEINLINE static bool convert(std::string* output, const bool& input)
[2171]403        {
404            if (input)
405              *output = "true";
406            else
407              *output = "false";
[2662]408            return true;
[2171]409        }
410    };
[1625]411
[2171]412    // std::string to bool
413    template <>
[7163]414    struct _UtilExport ConverterExplicit<std::string, bool>
[2171]415    {
[7163]416        static bool convert(bool* output, const std::string& input);
[2171]417    };
418}
419
[1052]420#endif /* _Convert_H__ */
Note: See TracBrowser for help on using the repository browser.