/* * ORXONOX - the hottest 3D action shooter ever to exist * > www.orxonox.net < * * * License notice: * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * Author: * Benjamin Grauer * Fabian 'x3n' Landau * Reto Grieder * Co-authors: * ... */ /*! @file @brief Definition and Implementation of the Convert class. */ #ifndef _Converter_H__ #define _Converter_H__ #include "UtilPrereqs.h" #include #include #include #include #include "Debug.h" // GCC generates warnings when implicitely casting from float to int for instance. // This is however exactly what convertValue does, so we need to suppress these warnings. // They only occur when using the ImplicitConversion template. #if ORXONOX_COMPILER == ORXONOX_COMPILER_GNUC # pragma GCC system_header #endif /////////////////////////////////////////////// // Static detection for conversion functions // /////////////////////////////////////////////// /* The idea to use the sizeof() operator on return functions to determine function existance is described in 'Modern C++ design' by Alexandrescu (2001). */ // disable warnings about possible loss of data #if ORXONOX_COMPILER == ORXONOX_COMPILER_MSVC # pragma warning(push) # pragma warning(disable:4244) #endif template class ImplicitConversion { private: ImplicitConversion(); ImplicitConversion(const ImplicitConversion&); ~ImplicitConversion(); // Gets chosen only if there is an implicit conversion from FromType to ToType. static char test(ToType); // Accepts any argument. Why do we not use a template? The reason is that with templates, // the function above is only taken iff it is an exact type match. But since we want to // check for implicit conversion, we have to use the ellipsis. static long long test(...); static FromType object; // helper object to handle private c'tor and d'tor public: // test(object) only has 'long long' return type iff the compiler doesn't choose test(...) enum { exists = (sizeof(test(object)) == sizeof(char)) }; }; #if ORXONOX_COMPILER == ORXONOX_COMPILER_MSVC # pragma warning(pop) #endif //////////////////////////////////// //// ACTUAL CONVERSION SEQUENCE //// //////////////////////////////////// /* There is a distinct priority when choosing the right conversion function: Overwrite: 1. (Partial) template specialisation of ConverterExplicit::convert() Fallbacks: 2. Any possible implicit conversion. This includes 'FooBar' --> 'int' if FooBar defines operator float(). 3. Global or member operators for stringstream when converting from or to std::string (or FROM const char*) 4. (Partial) template specialisation of ConverterFallback::convert() 5. Function that simply displays "Could not convert value" with type information obtained from typeid(). Notes: There has to be an exact type match when using template specialisations. Template specialisations can be defined after including this file. Any implicit cast function or iostream operator has to be declared BEFORE this file gets parsed. Defining your own functions: There are obviously 4 ways to specifiy a user defined conversion. What should I use? Usually, ConverterFallback fits quite well. You won't have to deal with the conversion from 'MyClass' to 'MyClass' by using another explicit template specialisation to avoid ambiguities. However if you want to overwrite an implicit conversion or an iostream operator, you really need to make use of ConverterExplicit. */ namespace { //! Little template that maps integers to entire types (Alexandrescu 2001) template struct Int2Type { }; } /////////////////// // No Conversion // /////////////////// // Default template. No conversion available at all. template struct ConverterFallback { static bool convert(ToType* output, const FromType& input) { COUT(2) << "Could not convert value of type " << typeid(FromType).name() << " to type " << typeid(ToType).name() << std::endl; return false; } }; /////////////////////// // ConverterFallback // /////////////////////// // Default template for stringstream template struct ConverterStringStream { static bool convert(ToType* output, const FromType& input) { return ConverterFallback::convert(output, input); } }; ///////////// // OStream // ///////////// namespace fallbackTemplates { template inline bool operator <<(std::ostream& outstream, const FromType& input) { std::string temp; if (ConverterFallback::convert(&temp, input)) { std::operator <<(outstream, temp); return true; } else return false; } } // template that evaluates whether we can convert to std::string via ostringstream template struct ConverterStringStream { static bool convert(std::string* output, const FromType& input) { using namespace fallbackTemplates; // this operator call only chooses fallbackTemplates::operator<< if there's no other fitting function std::ostringstream oss; if (oss << input) { (*output) = oss.str(); return true; } else return false; } }; ///////////// // IStream // ///////////// namespace fallbackTemplates { template inline bool operator >>(std::istream& instream, ToType& output) { return ConverterFallback ::convert(&output, static_cast(instream).str()); } } // template that evaluates whether we can convert from std::string via ostringstream template struct ConverterStringStream { static bool convert(ToType* output, const std::string& input) { using namespace fallbackTemplates; std::istringstream iss(input); // this operator call only chooses fallbackTemplates::operator>> if there's no other fitting function if (iss >> (*output)) { return true; } else return false; } }; /////////////////// // Implicit Cast // /////////////////// // implicit cast not possible, try stringstream conversion next template inline bool convertImplicitely(ToType* output, const FromType& input, ::Int2Type) { return ConverterStringStream::convert(output, input); } // We can cast implicitely template inline bool convertImplicitely(ToType* output, const FromType& input, ::Int2Type) { (*output) = static_cast(input); return true; } //////////////////////////////// // ConverterExplicit Fallback // //////////////////////////////// // Default template if no specialisation is available template struct ConverterExplicit { static bool convert(ToType* output, const FromType& input) { // Try implict cast and probe first. If a simple cast is not possible, it will not compile // We therefore have to out source it into another template function const bool probe = ImplicitConversion::exists; return convertImplicitely(output, input, ::Int2Type()); } }; ////////////////////// // Public Functions // ////////////////////// /** @brief Converts any value to any other as long as there exists a conversion. Otherwise, the conversion will generate a runtime warning and return false. For information about the different conversion methods (user defined too), see the section 'Actual conversion sequence' in this file above. */ template inline bool convertValue(ToType* output, const FromType& input) { return ConverterExplicit::convert(output, input); } // For compatibility reasons. The same, but with capital ConvertValue template inline bool ConvertValue(ToType* output, const FromType& input) { return convertValue(output, input); } // Calls convertValue and returns true if the conversion was successful. // Otherwise the fallback is used. /** @brief Converts any value to any other as long as there exists a conversion. Otherwise, the conversion will generate a runtime warning and return false. For information about the different conversion methods (user defined too), see the section 'Actual conversion sequence' in this file above. If the conversion doesn't succeed, 'fallback' is written to '*output'. @param fallback A default value that gets written to '*output' if there is no conversion. */ template inline bool convertValue(ToType* output, const FromType& input, const ToType& fallback) { if (convertValue(output, input)) return true; else { (*output) = fallback; return false; } } // for compatibility reason. (capital 'c' in ConvertValue) template inline bool ConvertValue(ToType* output, const FromType& input, const ToType& fallback) { return convertValue(output, input, fallback); } // Directly returns the converted value, even if the conversion was not successful. template inline ToType getConvertedValue(const FromType& input) { ToType output; convertValue(&output, input); return output; } // Directly returns the converted value, but uses the fallback on failure. template inline ToType getConvertedValue(const FromType& input, const ToType& fallback) { ToType output; convertValue(&output, input, fallback); return output; } // Like getConvertedValue, but the template argument order is in reverse. // That means you can call it exactly like static_cast(fromTypeValue). template inline ToType conversion_cast(const FromType& input) { ToType output; convertValue(&output, input); return output; } //////////////////////////////// // Special string conversions // //////////////////////////////// // Delegate conversion from const char* to std::string template struct ConverterExplicit { static bool convert(ToType* output, const char* input) { return convertValue(output, input); } }; // These conversions would exhibit ambiguous << or >> operators when using stringstream template <> struct ConverterExplicit { static bool convert(std::string* output, const char input) { *output = std::string(1, input); return true; } }; template <> struct ConverterExplicit { static bool convert(std::string* output, const unsigned char input) { *output = std::string(1, input); return true; } }; template <> struct ConverterExplicit { static bool convert(char* output, const std::string input) { if (input != "") *output = input[0]; else *output = '\0'; return true; } }; template <> struct ConverterExplicit { static bool convert(unsigned char* output, const std::string input) { if (input != "") *output = input[0]; else *output = '\0'; return true; } }; #endif /* _Convert_H__ */