-
Notifications
You must be signed in to change notification settings - Fork 149
/
ui_i18n.c
211 lines (165 loc) · 5.02 KB
/
ui_i18n.c
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
#include <stdint.h>
#include <string.h>
#include "sized_string.h"
#include "ui_i18n_decls.h"
#ifdef msgid
#error "msgid is already defined"
#endif
#ifdef msgstr
#error "msgstr is already defined"
#endif
#ifdef LANG_POSIX_LOCALE
#error "LANG_POSIX_LOCALE is already defined"
#endif
#ifdef LANG_WINDOWS_ID
#error "LANG_WINDOWS_ID is already defined"
#endif
#ifdef LANG_PRIORITY
#error "LANG_PRIORITY is already defined"
#endif
/***** Parsing localized strings *****/
#define msgid(x) curr_id = (STR_##x);
#define msgstr(x) \
localized_strings[_LANG_ID][curr_id].str = (uint8_t*)(x); \
localized_strings[_LANG_ID][curr_id].length = sizeof(x) - 1;
#define LANG_WINDOWS_ID(x)
#define LANG_POSIX_LOCALE(x)
#define LANG_PRIORITY(x)
static STRING canary = STRING_INIT("BUG. PLEASE REPORT.");
static void init_strings(STRING (*localized_strings)[STRS_MAX+1]) {
UI_LANG_ID i;
UI_STRING_ID j;
for(i = 0; i <= LANGS_MAX; i++) {
for(j = 0; j <= STRS_MAX; j++) {
localized_strings[i][j] = canary;
}
}
UI_STRING_ID curr_id = 0;
#include "ui_i18n.h"
}
#undef LANG_PRIORITY
#undef LANG_POSIX_LOCALE
#undef LANG_WINDOWS_ID
#undef msgstr
#undef msgid
STRING* ui_gettext(UI_LANG_ID lang, UI_STRING_ID string_id) {
static STRING localized_strings[LANGS_MAX+1][STRS_MAX+1];
static int ready = 0;
if(!ready) {
init_strings(localized_strings);
ready = 1;
}
if((lang > LANGS_MAX) || (string_id > STRS_MAX)) {
return &canary;
}
return &localized_strings[lang][string_id];
}
/***** Parsing detection by POSIX locale *****/
#define msgid(x)
#define msgstr(x)
#define LANG_WINDOWS_ID(x)
#define LANG_POSIX_LOCALE(x) posix_locales[_LANG_ID] = (x);
#define LANG_PRIORITY(x) priorities[_LANG_ID] = (x);
static void init_posix_locales(const char* posix_locales[], int8_t priorities[]) {
#include "ui_i18n.h"
}
#undef LANG_PRIORITY
#undef LANG_POSIX_LOCALE
#undef LANG_WINDOWS_ID
#undef msgstr
#undef msgid
UI_LANG_ID ui_guess_lang_by_posix_locale(const char* locale, UI_LANG_ID deflt) {
static const char* posix_locales[LANGS_MAX+1];
static int8_t priorities[LANGS_MAX+1];
static int ready = 0;
if(!ready) {
init_posix_locales(posix_locales, priorities);
ready = 1;
}
UI_LANG_ID i;
UI_LANG_ID found_lang = 0;
int8_t found_prio = INT8_MAX;
// Try detecting by full prefix match first.
for(i = 0; i <= LANGS_MAX; i++) {
const char* l = posix_locales[i];
if(!l) continue;
if(strstr(locale, l)) {
if(found_prio > priorities[i]) {
found_lang = i;
found_prio = priorities[i];
}
}
}
if(found_prio < INT8_MAX) {
return found_lang;
}
// It appears we haven't found exact language_territory
// match (e.g. zh_TW) for given locale. ,
// Try stripping territory off and search only by language part.
for(i = 0; i <= LANGS_MAX; i++) {
const char* l = posix_locales[i];
if(!l) continue;
char* sep = strchr(l, '_');
if(!sep) continue;
if(!strncmp(locale, l, sep-l)) {
if(found_prio > priorities[i]) {
found_lang = i;
found_prio = priorities[i];
}
}
}
return found_prio < INT8_MAX ? found_lang : deflt;
}
/***** Parsing detection by Windows language id *****/
#define msgid(x)
#define msgstr(x)
#define LANG_WINDOWS_ID(x) windows_lang_ids[_LANG_ID] = (x);
#define LANG_POSIX_LOCALE(x)
#define LANG_PRIORITY(x) priorities[_LANG_ID] = (x);
static void init_windows_lang_ids(uint16_t windows_lang_ids[], int8_t priorities[]) {
#include "ui_i18n.h"
}
#undef LANG_PRIORITY
#undef LANG_POSIX_LOCALE
#undef LANG_WINDOWS_ID
#undef msgstr
#undef msgid
UI_LANG_ID ui_guess_lang_by_windows_lang_id(uint16_t lang_id, UI_LANG_ID deflt) {
static uint16_t windows_lang_ids[LANGS_MAX+1];
static int8_t priorities[LANGS_MAX+1];
static int ready = 0;
if(!ready) {
init_windows_lang_ids(windows_lang_ids, priorities);
ready = 1;
}
UI_LANG_ID i;
UI_LANG_ID found_lang = 0;
int8_t found_prio = INT8_MAX;
// Try detecting by full match first, including sublanguage part.
for(i = 0; i <= LANGS_MAX; i++) {
uint16_t l = windows_lang_ids[i];
if(!l) continue;
if(l == lang_id) {
if(found_prio > priorities[i]) {
found_lang = i;
found_prio = priorities[i];
}
}
}
if(found_prio < INT8_MAX) {
return found_lang;
}
// It appears we haven't found exact id match.
// Try matching by the lower 8 bits, which contain language family part.
for(i = 0; i <= LANGS_MAX; i++) {
uint16_t l = windows_lang_ids[i];
if(!l) continue;
if((l&0xFF) == (lang_id&0xFF)) {
if(found_prio > priorities[i]) {
found_lang = i;
found_prio = priorities[i];
}
}
}
return found_prio < INT8_MAX ? found_lang : deflt;
}