forked from ixchow/kit
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathLoad.hpp
76 lines (64 loc) · 2.13 KB
/
Load.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
#pragma once
/*
* With the introduction of Mode, there is a need to synchronize resource loading
* among different compilation units.
*
* A Load< T > does this, by allowing you to write:
*
* //at global scope:
* Load< Mesh > main_mesh([]() -> const Mesh * {
* return &Meshes.get("Main");
* });
*
* //later:
* void GameMode::draw() {
* glBindVertexArray(main_mesh->vao);
* }
*
* Load<> is built on the add_load_function() call that adds a function to one of several lists of functions that are called after the OpenGL canvas is initialized.
*
* These functions are grouped by 'tags', which allow some sequencing of calls.
* (particularly, this is useful for loading large data blobs [e.g. "Meshes"] before looking up individual elements within them.)
*
*/
#include <functional>
#include <stdexcept>
namespace kit {
enum LoadTag : uint32_t {
LoadTagInit = 0, //used for loading mesh and texture blobs before default loading phase
LoadTagDefault = 1,
LoadTagLate = 2,
LoadTagCount = 3
};
void add_load_function(LoadTag tag, std::function< void() > const &fn);
void call_load_functions(); //called by main() after GL context created.
//work-around for MSVC not accepting this as a lambda:
template< typename T >
T const *new_T() { return new T; }
template< typename T >
struct Load {
//Constructing a Load< T > adds the passed function to the list of functions to call:
Load(LoadTag tag, const std::function< T const *() > &load_fn = new_T< T >) : value(nullptr) {
add_load_function(tag, [this,load_fn](){
this->value = load_fn();
if (!(this->value)) {
throw std::runtime_error("Loading failed.");
}
});
}
//Make a "Load< T >" behave like a "T const *":
explicit operator bool() { return value != nullptr; }
operator T const *() { return value; }
T const &operator*() { return *value; }
T const *operator->() { return value; }
T const *value;
};
//Load< void > just calls a function:
template< >
struct Load< void > {
//Constructing a Load< T > adds the passed function to the list of functions to call:
Load( LoadTag tag, const std::function< void() > &load_fn) {
add_load_function(tag, load_fn);
}
};
}