12 std::vector<std::string> items;
13 bool completed =
false;
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();
22 std::function<std::vector<std::string>(
const Property &)> list;
23 std::function<void(const Property &, const std::string &, AutoCompletion &)>
27 template <
class T>
bool expandList(std::vector<Property> &list, T &v);
29 bool expandList(std::vector<Property> &list, std::vector<T> &v);
34 std::shared_ptr<const Type> _type;
35 std::shared_ptr<const PropertyAttributes> _attributes;
39 void init(
const std::string &name,
40 const std::shared_ptr<PropertyAttributes> &attributes) {
41 static std::shared_ptr<Type> t = Type::create<T>();
45 _attributes = attributes;
47 static auto default_attributes = std::make_shared<PropertyAttributes>();
48 _attributes = default_attributes;
54 const std::shared_ptr<const PropertyAttributes> &attributes()
const {
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,
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;
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;
82 const std::shared_ptr<PropertyAttributes> &attributes) {
83 init<T>(name, attributes);
85 virtual void forEachObject(
void *context,
87 const Object *parent,
void *value)
const override {
88 ::forEachObject(context, f, parent, *(T *)value);
90 virtual void remove(
void *parent,
void *object)
const override {
91 removeObject(*(T *)parent,
object);
93 virtual bool canContainObjects()
const override {
94 return !std::is_same<decltype(::forEachObject(
nullptr,
nullptr,
nullptr,
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");
104 *(T *)ptr = v.value<T>();
106 virtual Variant toVariant(
const void *ptr)
const {
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;
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);
122 void *_value =
nullptr;
123 std::shared_ptr<const PropertyInfo> _info;
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 {
134 inline const std::shared_ptr<const PropertyAttributes> &attributes()
const {
135 return _info->attributes();
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);
142 inline void applySnapshot(
const std::shared_ptr<const SnapshotBase> &x) {
143 _info->applySnapshot(_value, x);
145 inline const std::type_index &typeId()
const {
146 return _info->type()->typeId();
148 inline bool tryToString(std::string &s)
const {
149 return _info->type()->tryToString(_value, s);
151 inline bool fromStringSupported()
const {
152 return _info->type()->fromStringSupported();
154 inline void fromStringOrThrow(
const std::string &s) {
155 _info->type()->fromStringOrThrow(_value, s);
157 inline void expand(std::vector<Property> &properties)
const {
158 _info->type()->properties(_value, properties);
160 inline void forEachObject(
void *context,
162 const Object *parent)
const {
163 _info->forEachObject(context, f, parent, _value);
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]);
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) {
178 if (typeId() !=
typeid(T)) {
179 throw std::runtime_error(
"property type mismatch");
181 return *
static_cast<T *
>(_value);
183 template <
class T>
inline const T &
get()
const {
185 if (typeId() !=
typeid(T)) {
186 throw std::runtime_error(
"property type mismatch");
188 return *
static_cast<const T *
>(_value);
190 inline void *valuePointer() {
194 inline const void *valuePointer()
const {
198 inline bool expandList(std::vector<Property> &list) {
199 return _info->expandList(list, _value);
201 static void checkUnlocked();
202 static void unlockScope(
int i);
205 template <
class T>
bool expandList(std::vector<Property> &list, T &v) {
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++) {
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; }
233 #define GENERATE_PROPERTY_ATTRIBUTES(type, ...) \ 235 struct Attr : PropertyAttributes { \ 237 DefaultPropertyAttributes<type>::initialize( \ 238 (PropertyAttributes *)this); \ 242 static auto attr = std::make_shared<Attr>(); \ 246 #define PROPERTY_FIRST_ARGUMENT_2(a, ...) a 247 #define PROPERTY_FIRST_ARGUMENT(...) PROPERTY_FIRST_ARGUMENT_2(__VA_ARGS__, ) 249 #define PROPERTY_SKIP_FIRST_2(a, ...) __VA_ARGS__ 250 #define PROPERTY_SKIP_FIRST(...) PROPERTY_SKIP_FIRST_2(__VA_ARGS__, 0) 252 #define PROPERTY(type, name, ...) \ 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]); \ 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)); \ 270 Property##_##name _##name##_property = \ 271 Property##_##name(this, &_property_##name); \ 274 inline const type &name() const { \ 275 Property::checkUnlocked(); \ 276 return _property_##name; \ 278 inline type &name() { \ 279 Property::checkUnlocked(); \ 280 return _property_##name; \ 282 inline void name(const type &value) { \ 283 Property::checkUnlocked(); \ 284 _property_##name = value; \