struct RemoveConst<const T>
{ typedef T Type; };
+
template<typename T>
struct RemoveReference
{ typedef T Type; };
struct RemoveReference<T &>
{ typedef T Type; };
+
+template<bool c, typename R>
+struct EnableIf;
+
+template<typename R>
+struct EnableIf<true, R>
+{ typedef R Yes; };
+
+template<typename R>
+struct EnableIf<false, R>
+{ typedef R No; };
+
+
+/**
+Common fragments used in SFINAE-based decider constructs.
+*/
+struct Sfinae
+{
+ struct Yes { char c[2]; };
+ struct No { char c; };
+
+ template<size_t s>
+ struct Evaluate
+ {
+ enum { value = (s==sizeof(Yes)) };
+ };
+};
+
} // namespace Msp
#endif
#include <sstream>
#include <string>
#include <stdexcept>
+#include <msp/core/meta.h>
#include "fmt.h"
namespace Msp {
// Generic operators using stringstream
template<typename T>
-void operator<<(LexicalConverter &c, const T &v)
+struct HasFormattedOutput: Sfinae
+{
+ static std::ostream &s;
+ static T &v;
+
+ /* The expression must depend on the template parameter, or the compiler
+ will give an error. */
+ template<typename U>
+ static Yes f(int (*)[sizeof(s<<HasFormattedOutput<U>::v)]);
+ template<typename U>
+ static No f(...);
+
+ enum { value = Evaluate<sizeof(f<T>(0))>::value };
+};
+
+template<typename T>
+struct HasFormattedInput: Sfinae
+{
+ static std::istream &s;
+ static T &v;
+
+ template<typename U>
+ static Yes f(int (*)[sizeof(s>>HasFormattedOutput<U>::v)]);
+ template<typename U>
+ static No f(...);
+
+ enum { value = Evaluate<sizeof(f<T>(0))>::value };
+};
+
+
+template<typename T>
+typename EnableIf<HasFormattedOutput<T>::value, void>::Yes
+operator<<(LexicalConverter &c, const T &v)
{
std::ostringstream ss;
ss<<c.get_fmt()<<v;
}
template<typename T>
-void operator>>(const LexicalConverter &c, T &v)
+typename EnableIf<HasFormattedInput<T>::value, void>::Yes
+operator>>(const LexicalConverter &c, T &v)
{
std::istringstream ss(c.get());
ss.setf(std::ios_base::fmtflags(0), std::ios_base::skipws);