forked from fnc12/Mitsoko
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathWireframe.hpp
229 lines (178 loc) · 7.29 KB
/
Wireframe.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
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
#ifndef __VIPER_WIREFRAME_H
#define __VIPER_WIREFRAME_H
#include "iOSutil/iOSutil.hpp"
#include "AndroidUtil/AndroidUtil.hpp"
#include <string>
#include <memory>
#include <experimental/optional>
#include <functional>
#include <iostream>
#include "Util.hpp"
//#include "Selfish.hpp"
//#define WIREFRAME_DECL(module,arg,ret) struct Wireframe:public Viper::Argumentable<arg,Wireframe>,Viper::Callbackable<ret,Wireframe>,Viper::WireframeBase,Selfish<Wireframe>
namespace Mitsoko{
namespace tuple_helper {
template<size_t N, class ...Args>
struct iterator {
template<class L>
void operator()(std::tuple<Args...> &t, L l) {
l(std::get<N>(t));
iterator<N - 1, Args...>()(t, l);
}
};
template<class ...Args>
struct iterator<0, Args...>{
template<class L>
void operator()(std::tuple<Args...> &t, L l) {
l(std::get<0>(t));
}
};
template<size_t N>
struct iterator<N> {
template<class L>
void operator()(std::tuple<> &t, L l) {
//..
}
};
}
/**
* This is how wireframe works:
* every wireframe has two main base classes:
* 1) Argumentable<A,W> is used to provide arguments to module.
* in `WIREFRAME_DECL` macro second argument must be a type module receives
* as an argument. Once argument specified one can override `virtual void
* init(A)` function in module's presenter. `init` function in presenter acts as a costructor.
* Second template argument must be Wireframe itself. It is used to make `Argumentable` superclass
* unique. Every module Argumentable superclass must be unique cause every `Argumentable<A,W>` has
* a `staticArgument` static optional variable. You have to assign it before creating a new module
* if you want to pass arguments to module. Example: `MyProfile::Wireframe::statusArgument()=5;`.
* If there is no need to pass an argument just pass void in WIREFRAME_DECL second argument or
* Argumentable<A,W> first template argument. You can still override `init()` function in presenter
* when your module has void argument.
* 2) Callbackable<C,W> is used to specify a callback mechanism for the module. It has `std::function<void(C)> staticCallback`
* static variable that must be assigned before module creating. This variable is `move`d into `callback` non-static member
* in `Callbackable` constructor and can be accessed at any time from presenter (`wireframe.callback(5);`)
* or from wireframe itself. Second template argument `W` is very important cause every Callbackable superclass
* must be unique for every module cause it has statis variables. The third argument in `WIREFRAME_DECL` macro
* must be a return type or void if there nothing to return. However you cann still fire a callback if module
* has `void` return type - callback just does not have argument (see `Callbackable<void,W>` specialization).
*/
template<class C,class W>
struct Callbackable {
typedef C return_type;
std::function<void(C)> callback;
static std::function<void(C)> staticCallback;
Callbackable():callback(std::move(staticCallback)){}
};
template<class C,class W>
std::function<void(C)> Callbackable<C, W>::staticCallback;
template<class W>
struct Callbackable<void,W> {
typedef void return_type;
std::function<void()> callback;
static std::function<void()> staticCallback;
Callbackable():callback(std::move(staticCallback)){}
};
template<class W>
std::function<void()> Callbackable<void, W>::staticCallback;
template<class A, class W>
struct Argumentable{
typedef A argument_type;
/**
* Tempopary storage for argument. Must be assigned from called module wireframe before
* switching to another module.
*/
static std::experimental::optional<argument_type> staticArgument;
};
template<class A, class W>
std::experimental::optional<A> Argumentable<A, W>::staticArgument;
template<class W>
struct Argumentable<void, W> {
typedef void argument_type;
};
template<class T>
struct Argument_t {
T t;
};
template<class T>
struct Callback_t {
T t;
};
struct WireframeBase {
const void *handle = nullptr;
template<class P, class T>
void processOpenArgument(Argument_t<T> a) {
P::staticArgument = std::move(a.t);
}
template<class P, class T>
void processOpenArgument(Callback_t<T> c) {
P::staticCallback = std::move(c.t);
}
template<class I, class A>
void open(const std::string &viewName, I i, A a) {
#ifdef __APPLE__
i(this->handle, viewName);
#else
a(this->handle, viewName);
#endif
}
template<class P, class I, class A, class ...Args>
void open(I i, A a, Args ...args) {
typedef std::tuple<Args...> tuple_t;
auto t = std::make_tuple(std::forward<Args>(args)...);
tuple_helper::iterator<std::tuple_size<tuple_t>::value - 1, Args...> it;
it(t, [=](auto &v){
this->processOpenArgument<P>(std::move(v));
});
this->open(P::view_type::viewName, i, a);
}
template<class I, class A>
void close(I i, A a) {
#ifdef __APPLE__
i(this->handle);
#else
a(this->handle);
#endif
}
};
struct NavigationPusher {
bool animated = true;
void operator()(const void *handle, const std::string &viewName);
};
struct NavigationPoper {
bool animated = true;
void operator()(const void *handle);
};
struct NavigationClassPoper {
std::string targetClassName;
bool animated = true;
NavigationClassPoper() = delete;
NavigationClassPoper(const std::string &target, bool animated = true);
void operator()(const void *handle);
};
struct FragmentStarter {
void operator()(const void *handle, const std::string &viewName);
};
struct ActivityStarter {
bool finishSelf = false;
std::shared_ptr<int> result;
ActivityStarter(bool finishSelf_ = false):finishSelf(finishSelf_){}
ActivityStarter(bool finishSelf_, int result_):finishSelf(finishSelf_),result(std::make_shared<int>(result_)){}
void operator()(const void *handle, const std::string &viewName);
};
struct ActivityCloser {
std::shared_ptr<int> result;
ActivityCloser();
ActivityCloser(int result);
void operator()(const void *handle);
};
template<class T>
Argument_t<T> argument(T t) {
return {t};
}
template<class T>
Callback_t<T> callback(T t) {
return {t};
}
}
#endif /* __VIPER_WIREFRAME_H */