-
-
Notifications
You must be signed in to change notification settings - Fork 161
/
Copy pathModule.hpp
142 lines (119 loc) · 3.13 KB
/
Module.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
#pragma once
#include <bitset>
#include <unordered_map>
#include <memory>
#include <functional>
#include "Utils/NonCopyable.hpp"
#include "Utils/TypeInfo.hpp"
namespace acid {
template<typename Base>
class ModuleFactory {
public:
class TCreateValue {
public:
std::function<std::unique_ptr<Base>()> create;
typename Base::Stage stage;
std::vector<TypeId> requires;
};
using TRegistryMap = std::unordered_map<TypeId, TCreateValue>;
virtual ~ModuleFactory() = default;
static TRegistryMap &Registry() {
static TRegistryMap impl;
return impl;
}
template<typename ... Args>
class Requires {
public:
std::vector<TypeId> Get() const {
std::vector<TypeId> requires;
(requires.emplace_back(TypeInfo<Base>::template GetTypeId<Args>()), ...);
return requires;
}
};
template<typename T>
class Registrar : public Base {
public:
/**
* Virtual deconstructor called from the engine to clear the instance pointer.
*/
virtual ~Registrar() {
if (static_cast<T *>(this) == moduleInstance)
moduleInstance = nullptr;
}
/**
* Gets the engines instance.
* @return The current module instance.
*/
static T *Get() { return moduleInstance; }
protected:
/**
* Creates a new module singleton instance and registers into the module registry map.
* @tparam Args Modules that will be initialized before this module.
* @return A dummy value in static initialization.
*/
template<typename ... Args>
static bool Register(typename Base::Stage stage, Requires<Args...> &&requires = {}) {
ModuleFactory::Registry()[TypeInfo<Base>::template GetTypeId<T>()] = {[]() {
moduleInstance = new T();
// The registrar does not own the instance, the engine does, we just hold a raw pointer for convenience.
return std::unique_ptr<Base>(moduleInstance);
}, stage, requires.Get()};
return true;
}
inline static T *moduleInstance = nullptr;
};
};
/**
* @brief A interface used for defining engine modules.
*/
class ACID_EXPORT Module : public ModuleFactory<Module>, NonCopyable {
public:
/**
* @brief Represents the stage where the module will be updated in the engine.
*/
enum class Stage : uint8_t {
Never, Always, Pre, Normal, Post, Render
};
using StageIndex = std::pair<Stage, TypeId>;
virtual ~Module() = default;
/**
* The update function for the module.
*/
virtual void Update() = 0;
};
template class ACID_EXPORT TypeInfo<Module>;
class ACID_EXPORT ModuleFilter {
public:
ModuleFilter() {
// Include all modules by default.
include.set();
}
template<typename T>
bool Check() const noexcept {
return include.test(TypeInfo<Module>::GetTypeId<T>());
}
bool Check(TypeId typeId) const noexcept {
return include.test(typeId);
}
template<typename T>
ModuleFilter &Exclude() noexcept {
include.reset(TypeInfo<Module>::GetTypeId<T>());
return *this;
}
template<typename T>
ModuleFilter &Include() noexcept {
include.set(TypeInfo<Module>::GetTypeId<T>());
return *this;
}
ModuleFilter &ExcludeAll() noexcept {
include.reset();
return *this;
}
ModuleFilter &IncludeAll() noexcept {
include.set();
return *this;
}
private:
std::bitset<64> include;
};
}