/* TRAIT.hpp - type handling and type detection helpers Copyright (C) Lumiera.org 2009, Hermann Vosseler 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., 675 Mass Ave, Cambridge, MA 02139, USA. */ #ifndef LUMIERA_META_TRAIT_H #define LUMIERA_META_TRAIT_H #include "lib/meta/util.hpp" #include #include #include #include #include namespace lumiera { namespace typelist { /** Trait template for detecting if a type can be converted to string. * For example, this allows to write specialisations with the help of * boost::enable_if */ template struct can_ToString { enum { value = boost::is_convertible::value }; }; /** Trait template for guarding \c lexical_cast<..> expressions. * Such an expression won't even compile for some types, because of * missing or ambiguous output operator(s). * Ideally, there would be some automatic detection (relying on the * existence of an operator<< for the given type. But I couldn't make * this work, so I fell back on just declaring types which are known * to work with lexical_cast to string * @note this compile-time trait can't predict if such an conversion * to string will be successful at runtime; indeed it may throw, * so you should additionally guard the invocation with try-catch! */ template struct can_lexical2string { enum { value = boost::is_arithmetic::value }; }; /** Extension to boost::reference_wrapper: * Allows additionally to re-bind to another reference, * almost like a pointer. For example this allows to cache * results returned from an API call by reference. * @warning potentially dangerous */ template class AssignableRefWrapper : public boost::reference_wrapper { typedef boost::reference_wrapper RefWrapper; public: explicit AssignableRefWrapper(TY& ref) : RefWrapper(ref) { } AssignableRefWrapper& operator= (RefWrapper const& o) { RefWrapper::operator= (o); return *this; } AssignableRefWrapper& operator= (TY& newRef) { (*this) = RefWrapper(newRef); return *this; } }; /** Type definition helper for pointer and reference types. * Allows to create a member field and to get the basic type * irrespective if the given type is plain, pointer or reference */ template struct RefTraits { typedef TY* pointer; typedef TY& reference; typedef TY value_type; typedef value_type member_type; }; template struct RefTraits { typedef TY* pointer; typedef TY& reference; typedef TY value_type; typedef pointer member_type; }; template struct RefTraits { typedef TY value_type; typedef const TY* pointer; typedef const TY& reference; typedef pointer member_type; }; template struct RefTraits { typedef TY* pointer; typedef TY& reference; typedef TY value_type; typedef AssignableRefWrapper member_type; }; }} // namespace lumiera::typelist #endif