2010-08-09 13:28:56 +00:00
|
|
|
#ifndef NALL_ANY_HPP
|
|
|
|
#define NALL_ANY_HPP
|
|
|
|
|
|
|
|
#include <typeinfo>
|
2012-06-18 10:13:51 +00:00
|
|
|
#include <nall/traits.hpp>
|
2010-08-09 13:28:56 +00:00
|
|
|
|
|
|
|
namespace nall {
|
2012-01-26 06:50:09 +00:00
|
|
|
struct any {
|
2010-08-09 13:28:56 +00:00
|
|
|
bool empty() const { return container; }
|
|
|
|
const std::type_info& type() const { return container ? container->type() : typeid(void); }
|
|
|
|
|
|
|
|
template<typename T> any& operator=(const T& value_) {
|
2012-01-26 06:50:09 +00:00
|
|
|
typedef typename type_if<
|
2010-08-09 13:28:56 +00:00
|
|
|
std::is_array<T>::value,
|
|
|
|
typename std::remove_extent<typename std::add_const<T>::type>::type*,
|
|
|
|
T
|
|
|
|
>::type auto_t;
|
|
|
|
|
|
|
|
if(type() == typeid(auto_t)) {
|
|
|
|
static_cast<holder<auto_t>*>(container)->value = (auto_t)value_;
|
|
|
|
} else {
|
|
|
|
if(container) delete container;
|
|
|
|
container = new holder<auto_t>((auto_t)value_);
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2012-04-20 12:48:09 +00:00
|
|
|
any() : container(nullptr) {}
|
|
|
|
~any() { if(container) delete container; }
|
|
|
|
template<typename T> any(const T& value_) : container(nullptr) { operator=(value_); }
|
2010-08-09 13:28:56 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
struct placeholder {
|
|
|
|
virtual const std::type_info& type() const = 0;
|
|
|
|
} *container;
|
|
|
|
|
|
|
|
template<typename T> struct holder : placeholder {
|
|
|
|
T value;
|
|
|
|
const std::type_info& type() const { return typeid(T); }
|
|
|
|
holder(const T& value_) : value(value_) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T> friend T any_cast(any&);
|
|
|
|
template<typename T> friend T any_cast(const any&);
|
|
|
|
template<typename T> friend T* any_cast(any*);
|
|
|
|
template<typename T> friend const T* any_cast(const any*);
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T> T any_cast(any &value) {
|
|
|
|
typedef typename std::remove_reference<T>::type nonref;
|
|
|
|
if(value.type() != typeid(nonref)) throw;
|
|
|
|
return static_cast<any::holder<nonref>*>(value.container)->value;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T> T any_cast(const any &value) {
|
|
|
|
typedef const typename std::remove_reference<T>::type nonref;
|
|
|
|
if(value.type() != typeid(nonref)) throw;
|
|
|
|
return static_cast<any::holder<nonref>*>(value.container)->value;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T> T* any_cast(any *value) {
|
2012-04-20 12:48:09 +00:00
|
|
|
if(!value || value->type() != typeid(T)) return nullptr;
|
2010-08-09 13:28:56 +00:00
|
|
|
return &static_cast<any::holder<T>*>(value->container)->value;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T> const T* any_cast(const any *value) {
|
2012-04-20 12:48:09 +00:00
|
|
|
if(!value || value->type() != typeid(T)) return nullptr;
|
2010-08-09 13:28:56 +00:00
|
|
|
return &static_cast<any::holder<T>*>(value->container)->value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|