1 /* lang.c - NeonServ v5.6
2 * Copyright (C) 2011-2012 Philipp Kreil (pk910)
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <http://www.gnu.org/licenses/>.
20 #include "mysqlConn.h"
23 #define DEFAULT_LANG_TAG "EN"
24 #define DEFAULT_LANG_NAME "English"
26 static struct language **langdict;
27 static struct language *lang_c;
30 langdict = calloc(MAXLANGUAGES, sizeof(*langdict));
37 void load_languages() {
40 printf_mysql_query("SELECT `lang`, `text` FROM `language` WHERE `ident` = 'name'");
42 while((row = mysql_fetch_row(res)) != NULL) {
43 load_language(row[0], row[1]);
47 static struct language* add_language(char *langtag, char *langname) {
49 for(cindex = 0; cindex < MAXLANGUAGES; cindex++) {
50 if(langdict[cindex] == NULL) break;
51 if(!strcmp(langdict[cindex]->langname, langname) || !strcmp(langdict[cindex]->langtag, langtag))
52 return langdict[cindex];
54 if(cindex == MAXLANGUAGES) return NULL;
55 struct language *lang = malloc(sizeof(*lang));
57 perror("malloc() failed");
60 lang->langtag = strdup(langtag);
61 lang->langname = strdup(langname);
62 struct language_table **entrys = calloc(27, sizeof(*entrys));
63 lang->entrys = entrys;
64 langdict[cindex] = lang;
68 static int get_entry_index(const char *ident) {
69 const char *underscore = strstr(ident, "_");
70 if(!underscore || !(underscore[1] >= 65 && underscore[1] <= 90)) return 26;
71 return (underscore[1] - 'A');
74 void load_language(char *tag, char *name) {
75 struct language *lang = get_language_by_tag(tag);
76 if(lang == get_default_language()) return;
80 struct language_table *entry, *next;
81 for(cindex = 0; cindex < 27; cindex++) {
82 for(entry = lang->entrys[cindex]; entry; entry = next) {
88 lang->entrys[cindex] = NULL;
91 lang = add_language(tag, name);
95 printf_mysql_query("SELECT `ident`, `text` FROM `language` WHERE `lang` = '%s' AND `ident` != 'name'", escape_string(tag));
97 while((row = mysql_fetch_row(res)) != NULL) {
98 register_language_string(lang, row[0], row[1]);
102 struct language* get_language_by_tag(char *tag) {
104 for(cindex = 0; cindex < MAXLANGUAGES; cindex++) {
105 if(langdict[cindex] == NULL) break;
106 if(!stricmp(langdict[cindex]->langtag, tag))
107 return langdict[cindex];
112 struct language* get_language_by_name(char *name) {
114 for(cindex = 0; cindex < MAXLANGUAGES; cindex++) {
115 if(langdict[cindex] == NULL) break;
116 if(!stricmp(langdict[cindex]->langname, name))
117 return langdict[cindex];
122 struct language* get_default_language() {
124 lang_c = add_language(DEFAULT_LANG_TAG, DEFAULT_LANG_NAME);
128 void register_default_language_table(const struct default_language_entry *msgtab) {
130 lang_c = add_language(DEFAULT_LANG_TAG, DEFAULT_LANG_NAME);
131 while(msgtab->ident) {
132 register_language_string(lang_c, msgtab->ident, msgtab->text);
137 void register_language_string(struct language *lang, char *ident, char *text) {
138 int cindex = get_entry_index(ident);
139 struct language_table *lang_entry;
140 for(lang_entry = lang->entrys[cindex]; lang_entry; lang_entry = lang_entry->next) {
141 if(!strcmp(lang_entry->ident, ident)) break;
144 lang_entry = malloc(sizeof(*lang_entry));
146 perror("malloc() failed");
149 lang_entry->ident = strdup(ident);
150 lang_entry->next = lang->entrys[cindex];
151 lang->entrys[cindex] = lang_entry;
153 free(lang_entry->text); //free old text (new one will be set below)
165 tmppos += sprintf(tmp + tmppos, "%s", b);
178 //unknown - just write it
179 tmppos += sprintf(tmp + tmppos, "$%c", a[1]);
184 lang_entry->text = strdup(tmp);
187 char *get_language_string(struct UserNode *user, const char* msg_ident) {
188 struct language* lang;
189 if(user && (user->flags & USERFLAG_ISAUTHED)) {
190 loadUserSettings(user);
191 lang = user->language;
194 int cindex = get_entry_index(msg_ident);
195 struct language_table* entry;
196 for(entry = lang->entrys[cindex]; entry; entry = entry->next) {
197 if(!strcmp(entry->ident, msg_ident))
200 if(lang == lang_c) return NULL;
201 for(entry = lang_c->entrys[cindex]; entry; entry = entry->next) {
202 if(!strcmp(entry->ident, msg_ident))
208 char *build_language_string(struct UserNode *user, char *buffer, const char *msg_ident, ...) {
209 char *formatStr = get_language_string(user, msg_ident);
210 if(!formatStr) return NULL;
212 buffer = (char *)malloc((MAXLEN+1) * sizeof(char));
214 perror("malloc() failed");
221 va_start(arg_list, msg_ident);
222 pos = vsnprintf(buffer, MAXLEN - 2, formatStr, arg_list);
224 if (pos < 0 || pos > (MAXLEN - 2)) pos = MAXLEN - 2;