-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathaliasenv_maker.h
142 lines (115 loc) · 3.45 KB
/
aliasenv_maker.h
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
#ifndef aliasenv_GUARD
#define aliasenv_GUARD
#include "defines.h"
#include "languages.h"
#include "writer.h"
#include <map>
#include <stack>
#include "aliasenv_empty.h"
namespace ctb
{
/**
* For general documentation see aliasenv_generator.h.
*
* This is a generalization of the aliasenv_generator class. aliasenv maker produces dynamicly constructable aliasenv, which can be used for creating general-purpose writers with custom alias tables.
*
* - T = tag (needed for distinguishing *static* alias tables)
* - L = language
* - R = report when not found (when handling strings on meta layer we may want misses
* */
template < typename T, typename L, bool R = true, typename F = aliasenv_empty>
class aliasenv_maker
{
protected:
T tag;
typedef map<string, string> aliastab_t;
static aliastab_t& aliases();
static stack<aliastab_t> my_aliases;
static void init();
public:
typedef aliasenv_maker<T,L,false,F> noreport;
typedef L language;
static string alias(const string& a, bool* success = NULL);
static void clear();
static string& access(const string&);
static string get_name();
static string get_child_name();
static void push();
static void pop();
static int depth();
};
struct empty_tag{};
template <typename T, typename L, bool R, typename F> stack<map<string, string>> aliasenv_maker<T,L,R,F>::my_aliases;
#define ADD(a,b) aliases.insert(aliastab_t::value_type(a,b))
template <typename T, typename L, bool R, typename F>
int aliasenv_maker<T,L,R,F>::depth()
{
return my_aliases.size();
}
template <typename T, typename L, bool R, typename F>
string aliasenv_maker<T,L,R,F>::get_child_name()
{
return F::get_name();
}
template <typename T, typename L, bool R, typename F>
string aliasenv_maker<T,L,R,F>::get_name()
{
return "maker";
}
template <typename T, typename L, bool R, typename F>
map<string, string>& aliasenv_maker<T,L,R,F>::aliases()
{
if(my_aliases.empty())
my_aliases.push(aliastab_t());
return my_aliases.top();
}
template <typename T, typename L, bool R, typename F>
void aliasenv_maker<T,L,R,F>::pop()
{
my_aliases.pop();
}
template <typename T, typename L, bool R, typename F>
void aliasenv_maker<T,L,R,F>::push()
{
my_aliases.push(aliastab_t());
}
template <typename T, typename L, bool R, typename F>
string aliasenv_maker<T,L,R,F>::alias(const string& a, bool* s)
{
auto itr = aliases().find(a);
if(itr == aliases().end())
{
bool mysuc;
string res = F::alias(a, &mysuc);
if(mysuc)
{
if(s != NULL)
*s = true;
return res;
}
if(s != NULL)
*s = false;
else if(R)
warn( string("warning: alias not found: ").append(a));
return "";
}
if(s != NULL)
*s = true;
return itr->second;
}
template <typename T, typename L, bool R, typename F>
void aliasenv_maker<T,L,R,F>::init()
{
}
template <typename T, typename L, bool R, typename F>
void aliasenv_maker<T,L,R,F>::clear()
{
aliases().clear();
}
template <typename T, typename L, bool R, typename F>
string& aliasenv_maker<T,L,R,F>::access(const string& str)
{
return aliases()[str];
}
};
#endif