2022-05-06 11:57:55 +08:00
|
|
|
package emu.grasscutter.utils;
|
|
|
|
|
|
|
|
import com.google.gson.JsonElement;
|
|
|
|
import com.google.gson.JsonObject;
|
|
|
|
import emu.grasscutter.Grasscutter;
|
|
|
|
|
|
|
|
import javax.annotation.Nullable;
|
|
|
|
import java.io.InputStream;
|
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
public final class Language {
|
|
|
|
private final JsonObject languageData;
|
|
|
|
private final Map<String, String> cachedTranslations = new HashMap<>();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a language instance from a code.
|
|
|
|
* @param langCode The language code.
|
|
|
|
* @return A language instance.
|
|
|
|
*/
|
|
|
|
public static Language getLanguage(String langCode) {
|
|
|
|
return new Language(langCode + ".json");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the translated value from the key while substituting arguments.
|
|
|
|
* @param key The key of the translated value to return.
|
|
|
|
* @param args The arguments to substitute.
|
|
|
|
* @return A translated value with arguments substituted.
|
|
|
|
*/
|
|
|
|
public static String translate(String key, Object... args) {
|
2022-05-07 09:52:10 +08:00
|
|
|
String translated = Grasscutter.getLanguage().get(key);
|
|
|
|
try {
|
|
|
|
return translated.formatted(args);
|
|
|
|
} catch (Exception exception) {
|
|
|
|
Grasscutter.getLogger().error("Failed to format string: " + key, exception);
|
|
|
|
return translated;
|
|
|
|
}
|
2022-05-06 11:57:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-05-07 12:50:10 +08:00
|
|
|
* creates a language instance.
|
2022-05-06 11:57:55 +08:00
|
|
|
* @param fileName The name of the language file.
|
|
|
|
*/
|
|
|
|
private Language(String fileName) {
|
|
|
|
@Nullable JsonObject languageData = null;
|
2022-05-07 12:50:10 +08:00
|
|
|
|
|
|
|
languageData = loadLanguage(fileName);
|
|
|
|
|
|
|
|
if (languageData == null) {
|
|
|
|
Grasscutter.getLogger().info("Now switch to default language");
|
|
|
|
languageData = loadDefaultLanguage();
|
|
|
|
}
|
|
|
|
|
|
|
|
assert languageData != null : "languageData is null";
|
|
|
|
this.languageData = languageData;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load default language file and creates a language instance.
|
|
|
|
* @return language data
|
|
|
|
*/
|
|
|
|
private JsonObject loadDefaultLanguage() {
|
|
|
|
var fileName = Grasscutter.getConfig().DefaultLanguage.toLanguageTag() + ".json";
|
|
|
|
return loadLanguage(fileName);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reads a file and creates a language instance.
|
|
|
|
* @param fileName The name of the language file.
|
|
|
|
* @return language data
|
|
|
|
*/
|
|
|
|
private JsonObject loadLanguage(String fileName) {
|
|
|
|
@Nullable JsonObject languageData = null;
|
2022-05-06 11:57:55 +08:00
|
|
|
|
|
|
|
try {
|
2022-05-07 09:04:39 +08:00
|
|
|
InputStream file = Grasscutter.class.getResourceAsStream("/languages/" + fileName);
|
2022-05-06 11:57:55 +08:00
|
|
|
languageData = Grasscutter.getGsonFactory().fromJson(Utils.readFromInputStream(file), JsonObject.class);
|
|
|
|
} catch (Exception exception) {
|
2022-05-07 12:50:10 +08:00
|
|
|
Grasscutter.getLogger().warn("Failed to load language file: " + fileName);
|
|
|
|
}
|
|
|
|
return languageData;
|
2022-05-06 11:57:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the value (as a string) from a nested key.
|
|
|
|
* @param key The key to look for.
|
|
|
|
* @return The value (as a string) from a nested key.
|
|
|
|
*/
|
|
|
|
public String get(String key) {
|
|
|
|
if(this.cachedTranslations.containsKey(key)) {
|
|
|
|
return this.cachedTranslations.get(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
String[] keys = key.split("\\.");
|
|
|
|
JsonObject object = this.languageData;
|
|
|
|
|
|
|
|
int index = 0;
|
2022-05-07 06:57:49 +08:00
|
|
|
String result = "This value does not exist. Please report this to the Discord: " + key;
|
2022-05-06 11:57:55 +08:00
|
|
|
|
|
|
|
while (true) {
|
|
|
|
if(index == keys.length) break;
|
|
|
|
|
|
|
|
String currentKey = keys[index++];
|
|
|
|
if(object.has(currentKey)) {
|
|
|
|
JsonElement element = object.get(currentKey);
|
|
|
|
if(element.isJsonObject())
|
|
|
|
object = element.getAsJsonObject();
|
|
|
|
else {
|
|
|
|
result = element.getAsString(); break;
|
|
|
|
}
|
|
|
|
} else break;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.cachedTranslations.put(key, result); return result;
|
|
|
|
}
|
|
|
|
}
|