12 inline void toString(
const std::string &v, std::string &s) { s = v; }
13 inline void toString(
const bool &v, std::string &s) {
20 inline void toString(
float v, std::string &s) { s = std::to_string(v); }
21 inline void toString(
double v, std::string &s) { s = std::to_string(v); }
22 inline void toString(
int v, std::string &s) { s = std::to_string(v); }
23 inline void toString(
long v, std::string &s) { s = std::to_string(v); }
24 inline void toString(
long long v, std::string &s) { s = std::to_string(v); }
25 inline void toString(
unsigned long v, std::string &s) { s = std::to_string(v); }
26 inline void toString(
unsigned long long v, std::string &s) {
27 s = std::to_string(v);
30 inline void fromString(std::string &v,
const std::string &s) { v = s; }
31 inline void fromString(
float &v,
const std::string &s) { v = std::stof(s); }
32 inline void fromString(
double &v,
const std::string &s) { v = std::stod(s); }
33 inline void fromString(
int &v,
const std::string &s) { v = std::stoi(s); }
34 inline void fromString(
long &v,
const std::string &s) { v = std::stol(s); }
35 inline void fromString(
long long &v,
const std::string &s) {
38 inline void fromString(
unsigned long &v,
const std::string &s) {
41 inline void fromString(
unsigned long long &v,
const std::string &s) {
44 inline void fromString(
bool &v,
const std::string &s) {
45 if (s ==
"true" || s ==
"1" || s ==
"True") {
49 if (s ==
"false" || s ==
"0" || s ==
"False") {
53 throw std::invalid_argument(
"s");
60 static inline bool tryToString(...) {
return false; }
62 static inline auto tryToString(
const T *v, std::string &s)
63 -> decltype(toString(*v, s),
bool()) {
68 static inline void fromStringOrThrow(...) {
69 throw std::invalid_argument(
"type");
72 static inline auto fromStringOrThrow(T *v,
const std::string &s)
73 -> decltype(fromString(*v, s)) {
77 static constexpr
bool toStringSupported(...) {
return false; }
79 static constexpr
auto toStringSupported(T *v)
80 -> decltype(toString(*v, *(std::string *)
nullptr),
true) {
84 static constexpr
bool fromStringSupported(...) {
return false; }
86 static constexpr
auto fromStringSupported(T *v)
87 -> decltype(fromString(*v, *(std::string *)
nullptr),
true) {
91 static constexpr
bool propertiesSupported(...) {
return false; }
93 static constexpr
auto propertiesSupported(T *v)
94 -> decltype(properties(*v, *(std::vector<Property> *)
nullptr),
true) {
98 inline void removeObject(...) {}
100 void removeObject(std::vector<std::shared_ptr<T>> &parent,
void *
object) {
101 for (
auto it = parent.begin(); it != parent.end();) {
102 if (it->get() == object) {
103 it = parent.erase(it);
105 removeObject(*it,
object);
110 void removeObject(
const std::shared_ptr<Object> &parent,
void *
object);
114 typedef std::function<bool(const std::shared_ptr<Object> &o)> SnapshotFilter;
116 uint64_t handleObjectId(
const Object *
object);
121 Handle(
const std::shared_ptr<T> &p =
nullptr) { reset(p); }
122 void reset(uint64_t
id) { _id = id; }
123 void reset(
const std::shared_ptr<T> &p =
nullptr) {
124 _id = handleObjectId(p ? (
Object *)p.get() :
nullptr);
126 void operator=(
const std::shared_ptr<T> &p) { reset(p); }
128 std::shared_ptr<T> resolve(
const std::shared_ptr<R> &root) {
132 std::shared_ptr<T> r;
133 root->recurse([&r,
this](
const std::shared_ptr<Object> &o) {
134 if (handleObjectId(o.get()) == _id) {
135 if (
auto x = std::dynamic_pointer_cast<T>(o)) {
142 const uint64_t id()
const {
return _id; }
144 template <
class T,
class S>
145 bool operator==(
const Handle<T> &h,
const std::shared_ptr<S> &p) {
149 return h.id() == p->id();
152 template <
class T,
class S>
153 bool operator==(
const std::shared_ptr<S> &p,
const Handle<T> &h) {
157 return h.id() == p->id();
160 template <
class T,
class S>
161 bool operator!=(
const Handle<T> &h,
const std::shared_ptr<S> &p) {
164 template <
class T,
class S>
165 bool operator!=(
const std::shared_ptr<S> &p,
const Handle<T> &h) {
170 auto forEachObject(
void *context,
172 const Object *parent,
const T &child) ->
173 typename std::enable_if<propertiesSupported((T *)
nullptr) ||
174 fromStringSupported((T *)
nullptr),
179 void forEachObject(
void *context,
184 void forEachObject(
void *context,
186 const Object *parent,
const std::shared_ptr<T> &v) {
187 forEachObject(context, f, parent, v.get());
191 void forEachObject(
void *context,
193 const Object *parent,
const std::vector<T> &v) {
195 forEachObject(context, f, parent, e);
201 forEachObject(
void *context,
void (*f)(
void *,
const Object *,
const Object *),
206 static void tryGetProperties(...) {}
208 static auto tryGetProperties(X *v, std::vector<Property> &props)
209 -> decltype(properties(*v, props)) {
210 properties(*v, props);
212 void getObjectProperties(
const std::shared_ptr<Object> &v,
213 std::vector<Property> &props);
215 static void tryGetProperties(std::shared_ptr<O> *v,
216 std::vector<Property> &props) {
217 getObjectProperties(*v, props);
223 std::type_index _type_id =
typeid(void);
224 std::shared_ptr<Type> _base;
225 bool _constructable =
false;
226 std::function<std::shared_ptr<void>()> _create;
227 std::string _category;
229 static void _registerType(
const std::shared_ptr<Type> &t);
231 template <
class T>
void _initConstructor(...) {
232 _create = []() -> std::shared_ptr<void> {
233 throw std::runtime_error(
234 "can't instantiate class without default constructor");
237 template <
class T>
void _initConstructor(decltype(
new T()) i) {
238 _create = []() -> std::shared_ptr<void> {
239 return std::static_pointer_cast<
void>(std::make_shared<T>());
241 _constructable =
true;
246 Type &operator=(
const Type &) =
delete;
250 const std::type_index &typeId()
const {
return _type_id; }
251 const std::string &name()
const {
return _name; }
252 const std::shared_ptr<Type> &base()
const {
return _base; }
253 bool constructable()
const {
return _constructable; }
254 const std::string &category()
const {
return _category; }
256 template <
class T> std::shared_ptr<T> instantiate()
const {
257 for (
const Type *t =
this; t; t = t->_base.get()) {
258 if (t->typeId() ==
typeid(T)) {
259 return std::static_pointer_cast<T>(_create());
265 static std::shared_ptr<Type> tryFind(
const std::string &name);
266 static std::shared_ptr<Type> tryFind(
const std::type_index &
id);
267 static std::shared_ptr<Type> find(
const std::string &name);
268 static std::shared_ptr<Type> find(
const std::type_index &
id);
269 template <
class T>
static std::shared_ptr<Type> find() {
270 return find(
typeid(T));
272 template <
class T>
static std::shared_ptr<Type> tryFind() {
273 return tryFind(
typeid(T));
276 std::vector<std::shared_ptr<Type>> list()
const;
279 static std::shared_ptr<Type>
280 global(
const std::string &name =
"",
281 const std::shared_ptr<Type> &base =
nullptr,
282 const std::string &category =
"") {
283 if (
auto ret = tryFind<T>()) {
286 auto ret = create<T>(name, base, category);
292 static std::shared_ptr<Type>
293 create(
const std::string &name =
"",
294 const std::shared_ptr<Type> &base =
nullptr,
295 const std::string &category =
"");
297 virtual bool tryToString(
const void *p, std::string &s)
const = 0;
298 virtual bool fromStringSupported()
const = 0;
299 virtual void fromStringOrThrow(
void *p,
const std::string &s)
const = 0;
300 virtual void properties(
const void *ptr,
301 std::vector<Property> &properties)
const = 0;
304 static bool _tryToPointerList(...) {
return false; }
306 static bool _tryToPointerList(
const std::vector<std::shared_ptr<T>> *list,
307 std::vector<std::shared_ptr<void>> &out) {
308 out.assign(list->begin(), list->end());
312 static bool _tryFromPointerList(...) {
return false; }
315 _tryFromPointerList(std::vector<std::shared_ptr<T>> *list,
316 const std::vector<std::shared_ptr<void>> &in) {
318 list->reserve(in.size());
320 list->push_back(std::static_pointer_cast<T>(v));
327 tryToPointerList(
const void *
object,
328 std::vector<std::shared_ptr<void>> &out)
const = 0;
330 tryFromPointerList(
void *
object,
331 const std::vector<std::shared_ptr<void>> &in)
const = 0;
337 TypeImpl(
const std::string &name,
const std::shared_ptr<Type> &base,
338 const std::string &category) {
341 _category = category;
342 _type_id =
typeid(T);
343 _initConstructor<T>(0);
345 virtual bool tryToString(
const void *p, std::string &s)
const override {
346 return ::tryToString((
const T *)p, s);
348 virtual bool fromStringSupported()
const {
349 return ::fromStringSupported((T *)
nullptr);
351 virtual void fromStringOrThrow(
void *value,
352 const std::string &s)
const override {
353 ::fromStringOrThrow((T *)value, s);
355 virtual void properties(
const void *ptr,
356 std::vector<Property> &props)
const override {
357 tryGetProperties((T *)ptr, props);
360 tryToPointerList(
const void *
object,
361 std::vector<std::shared_ptr<void>> &out)
const override {
362 return _tryToPointerList(static_cast<const T *>(
object), out);
364 virtual bool tryFromPointerList(
366 const std::vector<std::shared_ptr<void>> &in)
const override {
367 return _tryFromPointerList(static_cast<T *>(
object), in);
372 std::shared_ptr<Type> Type::create(
const std::string &name,
373 const std::shared_ptr<Type> &base,
374 const std::string &category) {
375 return std::shared_ptr<Type>(
new TypeImpl<T>(name, base, category));
378 #define DECLARE_TYPE_STRINGIFY_2(x) #x 379 #define DECLARE_TYPE_STRINGIFY(x) DECLARE_TYPE_STRINGIFY_2(x) 381 #define DECLARE_TYPE_IMPL(Derived, Base, ...) \ 382 static_assert(std::is_base_of<Base, Derived>::value, \ 383 "incorrect base class"); \ 384 static std::shared_ptr<Type> _##Base##_##Derived = Type::global<Derived>( \ 386 (typeid(Base) != typeid(Derived)) ? Type::find(typeid(Base)) : nullptr); 387 #define DECLARE_TYPE(Type, ...) DECLARE_TYPE_IMPL(Type, ##__VA_ARGS__, Type) 389 #define DECLARE_TYPE_C(Derived, Base, Category) \ 390 static_assert(std::is_base_of<Base, Derived>::value, \ 391 "incorrect base class"); \ 392 static std::shared_ptr<Type> _##Base##_##Derived = Type::global<Derived>( \ 393 #Derived, Type::find(typeid(Base)), DECLARE_TYPE_STRINGIFY_2(Category));