TAMSVIZ
Visualization and annotation tool for ROS
property.h
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #pragma once
5 
6 #include "type.h"
7 #include "variant.h"
8 
9 #include <limits>
10 
12  std::vector<std::string> items;
13  bool completed = false;
14 };
15 
17  double min = std::numeric_limits<double>::quiet_NaN();
18  double max = std::numeric_limits<double>::quiet_NaN();
19  double step_scale = std::numeric_limits<double>::quiet_NaN();
20  double wrap = false;
21  bool hidden = false;
22  std::function<std::vector<std::string>(const Property &)> list;
23  std::function<void(const Property &, const std::string &, AutoCompletion &)>
24  complete;
25 };
26 
27 template <class T> bool expandList(std::vector<Property> &list, T &v);
28 template <class T>
29 bool expandList(std::vector<Property> &list, std::vector<T> &v);
30 
31 class PropertyInfo {
32 private:
33  std::string _name;
34  std::shared_ptr<const Type> _type;
35  std::shared_ptr<const PropertyAttributes> _attributes;
36 
37 protected:
38  template <class T>
39  void init(const std::string &name,
40  const std::shared_ptr<PropertyAttributes> &attributes) {
41  static std::shared_ptr<Type> t = Type::create<T>();
42  _type = t;
43  _name = name;
44  if (attributes) {
45  _attributes = attributes;
46  } else {
47  static auto default_attributes = std::make_shared<PropertyAttributes>();
48  _attributes = default_attributes;
49  }
50  }
51  PropertyInfo() {}
52 
53 public:
54  const std::shared_ptr<const PropertyAttributes> &attributes() const {
55  return _attributes;
56  }
57  const std::string &name() const { return _name; }
58  const std::shared_ptr<const Type> &type() const { return _type; }
59  virtual void forEachObject(void *context,
60  void (*f)(void *, const Object *, const Object *),
61  const Object *parent, void *value) const = 0;
62  virtual void remove(void *parent, void *object) const = 0;
63  virtual bool canContainObjects() const = 0;
64  virtual Variant serialize(const void *ptr) const = 0;
65  virtual void deserialize(void *ptr, const Variant &v) const = 0;
66  virtual void assign(void *ptr, const Variant &v) const = 0;
67  virtual Variant toVariant(const void *ptr) const = 0;
68  virtual std::shared_ptr<const SnapshotBase>
69  save(const void *value, const std::shared_ptr<const SnapshotBase> &x,
70  const SnapshotFilter &filter) const = 0;
71  virtual void
72  applySnapshot(void *value,
73  const std::shared_ptr<const SnapshotBase> &x) const = 0;
74  virtual bool expandList(std::vector<Property> &list,
75  const void *ptr) const = 0;
76 };
77 
78 template <class T> class PropertyInfoImpl : public PropertyInfo {
79 
80 public:
81  PropertyInfoImpl(const std::string &name,
82  const std::shared_ptr<PropertyAttributes> &attributes) {
83  init<T>(name, attributes);
84  }
85  virtual void forEachObject(void *context,
86  void (*f)(void *, const Object *, const Object *),
87  const Object *parent, void *value) const override {
88  ::forEachObject(context, f, parent, *(T *)value);
89  }
90  virtual void remove(void *parent, void *object) const override {
91  removeObject(*(T *)parent, object);
92  }
93  virtual bool canContainObjects() const override {
94  return !std::is_same<decltype(::forEachObject(nullptr, nullptr, nullptr,
95  *(T *)nullptr)),
96  StopObjectRecursionTag>::value;
97  }
98  virtual Variant serialize(const void *value) const override;
99  virtual void deserialize(void *value, const Variant &v) const override;
100  virtual void assign(void *ptr, const Variant &v) const {
101  if (v.type() != typeid(T)) {
102  throw std::runtime_error("type mismatch");
103  }
104  *(T *)ptr = v.value<T>();
105  }
106  virtual Variant toVariant(const void *ptr) const {
107  return Variant(*(T *)ptr);
108  }
109  virtual std::shared_ptr<const SnapshotBase>
110  save(const void *value, const std::shared_ptr<const SnapshotBase> &x,
111  const SnapshotFilter &filter) const override;
112  virtual void
113  applySnapshot(void *value,
114  const std::shared_ptr<const SnapshotBase> &x) const override;
115  virtual bool expandList(std::vector<Property> &list,
116  const void *ptr) const override {
117  return ::expandList(list, *(T *)ptr);
118  }
119 };
120 
121 class Property {
122  void *_value = nullptr;
123  std::shared_ptr<const PropertyInfo> _info;
124 
125 public:
126  Property() {}
127  template <class T>
128  inline Property(const std::shared_ptr<PropertyInfo> &info, T *value)
129  : _info(info), _value(value) {}
130  inline const char *name() const { return _info->name().c_str(); }
131  inline const std::shared_ptr<const PropertyInfo> &info() const {
132  return _info;
133  }
134  inline const std::shared_ptr<const PropertyAttributes> &attributes() const {
135  return _info->attributes();
136  }
137  inline std::shared_ptr<const SnapshotBase>
138  save(const std::shared_ptr<const SnapshotBase> &x,
139  const SnapshotFilter &filter) const {
140  return _info->save(_value, x, filter);
141  }
142  inline void applySnapshot(const std::shared_ptr<const SnapshotBase> &x) {
143  _info->applySnapshot(_value, x);
144  }
145  inline const std::type_index &typeId() const {
146  return _info->type()->typeId();
147  }
148  inline bool tryToString(std::string &s) const {
149  return _info->type()->tryToString(_value, s);
150  }
151  inline bool fromStringSupported() const {
152  return _info->type()->fromStringSupported();
153  }
154  inline void fromStringOrThrow(const std::string &s) {
155  _info->type()->fromStringOrThrow(_value, s);
156  }
157  inline void expand(std::vector<Property> &properties) const {
158  _info->type()->properties(_value, properties);
159  }
160  inline void forEachObject(void *context,
161  void (*f)(void *, const Object *, const Object *),
162  const Object *parent) const {
163  _info->forEachObject(context, f, parent, _value);
164  }
165  inline void remove(void *object) { _info->remove(_value, object); }
166  inline bool canContainObjects() const { return _info->canContainObjects(); }
167  inline Variant serialize() const { return _info->serialize(_value); }
168  inline void deserialize(const Variant &v) { _info->deserialize(_value, v); }
169  std::string displayName() const {
170  std::string n = name();
171  n.at(0) = std::toupper(n[0]);
172  return n;
173  }
174  inline void assign(const Variant &variant) { _info->assign(_value, variant); }
175  inline Variant read() const { return _info->toVariant(_value); }
176  template <class T> inline void set(const T &v) {
177  checkUnlocked();
178  if (typeId() != typeid(T)) {
179  throw std::runtime_error("property type mismatch");
180  }
181  return *static_cast<T *>(_value);
182  }
183  template <class T> inline const T &get() const {
184  checkUnlocked();
185  if (typeId() != typeid(T)) {
186  throw std::runtime_error("property type mismatch");
187  }
188  return *static_cast<const T *>(_value);
189  }
190  inline void *valuePointer() {
191  checkUnlocked();
192  return _value;
193  }
194  inline const void *valuePointer() const {
195  checkUnlocked();
196  return _value;
197  }
198  inline bool expandList(std::vector<Property> &list) {
199  return _info->expandList(list, _value);
200  }
201  static void checkUnlocked();
202  static void unlockScope(int i);
203 };
204 
205 template <class T> bool expandList(std::vector<Property> &list, T &v) {
206  return false;
207 }
208 template <class T>
209 bool expandList(std::vector<Property> &list, std::vector<T> &v) {
210  static auto attr = std::make_shared<PropertyAttributes>();
211  for (size_t i = 0; i < v.size(); i++) {
212  list.push_back(Property(
213  std::make_shared<PropertyInfoImpl<T>>(std::to_string(i), attr), &v[i]));
214  }
215  return true;
216 }
217 
218 template <class T> class PropertyList {
219  T *_begin, *_end;
220 
221 public:
222  PropertyList(T *begin, T *end) : _begin(begin), _end(end) {}
223  T *begin() const { return _begin; }
224  T *end() const { return _end; }
225  T &operator[](size_t i) const { return _begin[i]; }
226  size_t size() const { return _end - _begin; }
227 };
228 
229 template <class T> struct DefaultPropertyAttributes {
230  static inline void initialize(PropertyAttributes *attributes) {}
231 };
232 
233 #define GENERATE_PROPERTY_ATTRIBUTES(type, ...) \
234  []() { \
235  struct Attr : PropertyAttributes { \
236  Attr() { \
237  DefaultPropertyAttributes<type>::initialize( \
238  (PropertyAttributes *)this); \
239  __VA_ARGS__; \
240  } \
241  }; \
242  static auto attr = std::make_shared<Attr>(); \
243  return attr; \
244  }()
245 
246 #define PROPERTY_FIRST_ARGUMENT_2(a, ...) a
247 #define PROPERTY_FIRST_ARGUMENT(...) PROPERTY_FIRST_ARGUMENT_2(__VA_ARGS__, )
248 
249 #define PROPERTY_SKIP_FIRST_2(a, ...) __VA_ARGS__
250 #define PROPERTY_SKIP_FIRST(...) PROPERTY_SKIP_FIRST_2(__VA_ARGS__, 0)
251 
252 #define PROPERTY(type, name, ...) \
253 private: \
254  type _property_##name = type(PROPERTY_FIRST_ARGUMENT(__VA_ARGS__)); \
255  struct Property##_##name { \
256  template <class O, class T> \
257  inline Property##_##name(O *object, T *value) { \
258  static std::string xname = []() { \
259  std::string s = #name; \
260  s[0] = std::toupper(s[0]); \
261  return s; \
262  }(); \
263  static auto info = \
264  std::shared_ptr<PropertyInfo>(new PropertyInfoImpl<T>( \
265  xname, GENERATE_PROPERTY_ATTRIBUTES( \
266  type, PROPERTY_SKIP_FIRST(__VA_ARGS__)))); \
267  object->addProperty(Property(info, value)); \
268  } \
269  }; \
270  Property##_##name _##name##_property = \
271  Property##_##name(this, &_property_##name); \
272  \
273 public: \
274  inline const type &name() const { \
275  Property::checkUnlocked(); \
276  return _property_##name; \
277  } \
278  inline type &name() { \
279  Property::checkUnlocked(); \
280  return _property_##name; \
281  } \
282  inline void name(const type &value) { \
283  Property::checkUnlocked(); \
284  _property_##name = value; \
285  }
Definition: object.h:8
Definition: variant.h:9