Skip to content

Commit 85be6af

Browse files
committedJan 18, 2022
Add menu option for language translations (fix #136)
1 parent 5a4f154 commit 85be6af

File tree

4 files changed

+103
-14
lines changed

4 files changed

+103
-14
lines changed
 

‎bin/language_scan.js

+54-12
Original file line numberDiff line numberDiff line change
@@ -2,22 +2,30 @@
22
/* Scans for strings that may be in English in each app, and
33
outputs a list of strings that have been found.
44
5-
Early work towards internationalisation.
6-
See https://github.com/espruino/BangleApps/issues/136
5+
See https://github.com/espruino/BangleApps/issues/1311
76
*/
87

8+
var IGNORE_STRINGS = [
9+
"5x5",
10+
"5x9Numeric7Seg",
11+
"Vector"
12+
];
13+
914
var BASEDIR = __dirname+"/../";
1015
Espruino = require(BASEDIR+"core/lib/espruinotools.js");
1116
var fs = require("fs");
12-
1317
var APPSDIR = BASEDIR+"apps/";
18+
1419
function ERROR(s) {
1520
console.error("ERROR: "+s);
1621
process.exit(1);
1722
}
1823
function WARN(s) {
1924
console.log("Warning: "+s);
2025
}
26+
function log(s) {
27+
console.log(s);
28+
}
2129

2230
var appsFile, apps;
2331
try {
@@ -39,31 +47,65 @@ function isNotString(s) {
3947
if (s.endsWith(".json") || s.endsWith(".img")) return true; // a filename
4048
if (s.endsWith("=")) return true; // probably base64
4149
if (s.startsWith("BTN")) return true; // button name
50+
if (IGNORE_STRINGS.includes(s)) return true; // one we know to ignore
4251
return false;
4352
}
4453

45-
var textStrings = [];
54+
// A string that *could* be translated?
55+
var untranslatedStrings = [];
56+
// Strings that are marked with 'LANG'
57+
var translatedStrings = [];
4658

47-
console.log("Scanning...");
59+
console.log("Scanning apps...");
4860
apps.forEach((app,appIdx) => {
4961
var appDir = APPSDIR+app.id+"/";
5062
app.storage.forEach((file) => {
5163
if (!file.url || !file.name.endsWith(".js")) return;
5264
var fileContents = fs.readFileSync(appDir+file.url).toString();
5365
var lex = Espruino.Core.Utils.getLexer(fileContents);
66+
var lastIdx = 0;
5467
var tok = lex.next();
5568
while (tok!==undefined) {
69+
var previousString = fileContents.substring(lastIdx, tok.startIdx);
5670
if (tok.type=="STRING") {
57-
if (!isNotString(tok.value)) {
58-
//console.log(tok.str);
59-
if (!textStrings.includes(tok.value))
60-
textStrings.push(tok.value);
71+
if (previousString.includes("/*LANG*/")) { // translated!
72+
if (!translatedStrings.includes(tok.value))
73+
translatedStrings.push(tok.value);
74+
} else { // untranslated - potential to translate?
75+
if (!isNotString(tok.value)) {
76+
if (!untranslatedStrings.includes(tok.value))
77+
untranslatedStrings.push(tok.value);
78+
}
6179
}
6280
}
81+
lastIdx = tok.endIdx;
6382
tok = lex.next();
6483
}
6584
});
6685
});
67-
console.log("Done");
68-
textStrings.sort();
69-
console.log(textStrings.join("\n"));
86+
untranslatedStrings.sort();
87+
translatedStrings.sort();
88+
89+
var report = "";
90+
/* // too many! don't output these
91+
log("Possible English Strings that could be translated");
92+
log("=================================================================");
93+
log("");
94+
log("Add these to IGNORE_STRINGS if the don't make sense...");
95+
log("");
96+
log(untranslatedStrings.map(s=>JSON.stringify(s)).join(",\n"));*/
97+
log("");
98+
99+
var languages = JSON.parse(fs.readFileSync(BASEDIR+"/lang/index.json").toString());
100+
languages.forEach(language => {
101+
console.log("Scanning "+language.code);
102+
log(language.code);
103+
log("==========");
104+
var translations = JSON.parse(fs.readFileSync(BASEDIR+"/lang/"+language.url).toString());
105+
translatedStrings.forEach(str => {
106+
if (!translations.GLOBAL[str])
107+
console.log(`Missing translation for ${JSON.stringify(str)}`);
108+
});
109+
log("");
110+
});
111+
console.log("Done.");

‎index.html

+5
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,11 @@ <h3>Settings</h3>
141141
<input type="checkbox" id="settings-settime">
142142
<i class="form-icon"></i> Always update time when we connect
143143
</label>
144+
<div class="form-group">
145+
<select class="form-select form-inline" id="settings-lang" style="width: 10em">
146+
<option value="">None (English)</option>
147+
</select>&nbsp;&nbsp;<span>Translations (<a href="https://github.com/espruino/BangleApps/issues/1311" target="_blank">BETA - more info</a>)</span>
148+
</div>
144149
<button class="btn" id="defaultsettings">Default settings</button>
145150
</div>
146151
</div>

‎lang/en_GB.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
11
{
22
"//":"British English language translations - the default strings in apps are all english anyway, so no need to have translations for most things",
33
"GLOBAL": {
4-
"//":"Translations that apply for all apps",
4+
"//":"Translations that apply for all apps"
55
},
66
"alarm": {
7-
"//":"App-specific overrides",
7+
"//":"App-specific overrides"
88
}
99
}

‎loader.js

+42
Original file line numberDiff line numberDiff line change
@@ -164,6 +164,48 @@ window.addEventListener('load', (event) => {
164164
showToast("App Install failed, "+err,"error");
165165
});
166166
});
167+
168+
// Load language list
169+
httpGet("lang/index.json").then(languagesJSON=>{
170+
var languages;
171+
try {
172+
languages = JSON.parse(languagesJSON);
173+
} catch(e) {
174+
console.error("lang/index.json Corrupted", e);
175+
}
176+
177+
function reloadLanguage() {
178+
LANGUAGE = undefined;
179+
if (SETTINGS.language) {
180+
var language = languages.find(l=>l.code==SETTINGS.language);
181+
if (language) {
182+
var langURL = "lang/"+language.url;
183+
httpGet(langURL).then(languageJSON=>{
184+
try {
185+
LANGUAGE = JSON.parse(languageJSON);
186+
console.log(`${langURL} loaded successfully`);
187+
} catch(e) {
188+
console.error(`${langURL} Corrupted`, e);
189+
}
190+
});
191+
} else {
192+
console.error(`Language code ${JSON.stringify(SETTINGS.language)} not found in lang/index.json`);
193+
}
194+
}
195+
}
196+
197+
var selectLang = document.getElementById("settings-lang");
198+
console.log(languages);
199+
languages.forEach(lang => {
200+
selectLang.innerHTML += `<option value="${lang.code}" ${SETTINGS.language==lang.code?"selected":""}>${lang.name} (${lang.code})</option>`;
201+
});
202+
selectLang.addEventListener("change",event=>{
203+
SETTINGS.language = event.target.value;
204+
reloadLanguage();
205+
saveSettings();
206+
});
207+
reloadLanguage();
208+
});
167209
});
168210

169211
function onAppJSONLoaded() {

0 commit comments

Comments
 (0)