1
0
mirror of https://github.com/TomHarte/CLK.git synced 2024-06-26 10:29:31 +00:00
CLK/Reflection/Struct.h

88 lines
2.2 KiB
C
Raw Normal View History

//
// Struct.h
// Clock Signal
//
// Created by Thomas Harte on 06/03/2020.
// Copyright © 2020 Thomas Harte. All rights reserved.
//
#ifndef Struct_h
#define Struct_h
#include <string>
2020-03-09 01:23:29 +00:00
#include <typeindex>
#include <typeinfo>
#include <unordered_map>
#include <vector>
namespace Reflection {
2020-03-09 01:23:29 +00:00
template <typename Owner> class Struct {
public:
2020-03-09 01:23:29 +00:00
template <typename Type> const Type *get(const std::string &name) {
const auto iterator = contents_.find(name);
if(iterator == contents_.end()) return nullptr;
return reinterpret_cast<Type *>(reinterpret_cast<uint8_t *>(this) + iterator->second.offset);
}
template <typename Type> void set(const std::string &name, const Type &value) {
2020-03-09 01:23:29 +00:00
const auto iterator = contents_.find(name);
if(iterator == contents_.end()) return;
*reinterpret_cast<Type *>(reinterpret_cast<uint8_t *>(this) + iterator->second.offset) = value;
}
2020-03-09 01:23:29 +00:00
const std::type_info *type_of(const std::string &name) {
const auto iterator = contents_.find(name);
if(iterator == contents_.end()) return nullptr;
return iterator->second.type;
}
std::vector<std::string> all_keys() {
std::vector<std::string> keys;
for(const auto &pair: contents_) {
keys.push_back(pair.first);
}
return keys;
}
protected:
2020-03-09 01:23:29 +00:00
/*
This interface requires reflective structs to declare all fields;
specifically they should call:
declare_field(&field1, "field1");
declare_field(&field2, "field2");
Fields are registered in class storage. So callers can use needs_declare()
to determine whether a class of this type has already established the
reflective fields.
*/
template <typename Type> void declare(Type *t, const std::string &name) {
2020-03-09 01:23:29 +00:00
contents_.emplace(
std::make_pair(
name,
Field(typeid(Type), reinterpret_cast<uint8_t *>(t) - reinterpret_cast<uint8_t *>(this))
));
}
/*!
@returns @c true if this
*/
bool needs_declare() {
return !contents_.size();
}
private:
2020-03-09 01:23:29 +00:00
struct Field {
const std::type_info *type;
ssize_t offset;
Field(const std::type_info &type, ssize_t offset) :
type(&type), offset(offset) {}
};
static inline std::unordered_map<std::string, Field> contents_;
};
}
#endif /* Struct_h */