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;
|
2022-05-10 20:33:45 +08:00
|
|
|
import emu.grasscutter.game.player.Player;
|
2022-05-06 11:57:55 +08:00
|
|
|
|
|
|
|
import javax.annotation.Nullable;
|
|
|
|
import java.io.InputStream;
|
2022-05-09 17:30:51 +08:00
|
|
|
import java.util.concurrent.ConcurrentHashMap;
|
2022-05-06 11:57:55 +08:00
|
|
|
import java.util.Map;
|
|
|
|
|
2022-05-11 12:30:07 +08:00
|
|
|
import static emu.grasscutter.Configuration.*;
|
|
|
|
|
2022-05-06 11:57:55 +08:00
|
|
|
public final class Language {
|
2022-05-11 08:00:47 +08:00
|
|
|
private static final Map<String, Language> cachedLanguages = new ConcurrentHashMap<>();
|
|
|
|
|
2022-05-06 11:57:55 +08:00
|
|
|
private final JsonObject languageData;
|
2022-05-10 20:33:45 +08:00
|
|
|
private final String languageCode;
|
2022-05-09 17:30:51 +08:00
|
|
|
private final Map<String, String> cachedTranslations = new ConcurrentHashMap<>();
|
2022-05-06 11:57:55 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a language instance from a code.
|
|
|
|
* @param langCode The language code.
|
|
|
|
* @return A language instance.
|
|
|
|
*/
|
|
|
|
public static Language getLanguage(String langCode) {
|
2022-05-09 17:30:51 +08:00
|
|
|
if (cachedLanguages.containsKey(langCode)) {
|
|
|
|
return cachedLanguages.get(langCode);
|
|
|
|
}
|
|
|
|
|
2022-05-11 12:30:07 +08:00
|
|
|
var fallbackLanguageCode = Utils.getLanguageCode(FALLBACK_LANGUAGE);
|
|
|
|
var description = getLanguageFileDescription(langCode, fallbackLanguageCode);
|
2022-05-11 08:00:47 +08:00
|
|
|
var actualLanguageCode = description.getLanguageCode();
|
2022-05-10 20:33:45 +08:00
|
|
|
|
2022-05-11 08:00:47 +08:00
|
|
|
Language languageInst;
|
|
|
|
if (description.getLanguageFile() != null) {
|
|
|
|
languageInst = new Language(description);
|
2022-05-10 20:33:45 +08:00
|
|
|
cachedLanguages.put(actualLanguageCode, languageInst);
|
2022-05-11 08:00:47 +08:00
|
|
|
} else {
|
2022-05-10 20:33:45 +08:00
|
|
|
languageInst = cachedLanguages.get(actualLanguageCode);
|
|
|
|
cachedLanguages.put(langCode, languageInst);
|
|
|
|
}
|
|
|
|
|
2022-05-09 17:30:51 +08:00
|
|
|
return languageInst;
|
2022-05-06 11:57:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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);
|
2022-05-08 06:12:53 +08:00
|
|
|
|
2022-05-07 09:52:10 +08:00
|
|
|
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-10 20:33:45 +08:00
|
|
|
/**
|
|
|
|
* Returns the translated value from the key while substituting arguments.
|
|
|
|
* @param player Target player
|
|
|
|
* @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(Player player, String key, Object... args) {
|
|
|
|
if (player == null) {
|
|
|
|
return translate(key, args);
|
|
|
|
}
|
|
|
|
|
|
|
|
var langCode = Utils.getLanguageCode(player.getAccount().getLocale());
|
|
|
|
String translated = Grasscutter.getLanguage(langCode).get(key);
|
|
|
|
|
|
|
|
try {
|
|
|
|
return translated.formatted(args);
|
|
|
|
} catch (Exception exception) {
|
|
|
|
Grasscutter.getLogger().error("Failed to format string: " + key, exception);
|
|
|
|
return translated;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get language code
|
|
|
|
*/
|
|
|
|
public String getLanguageCode() {
|
|
|
|
return languageCode;
|
|
|
|
}
|
|
|
|
|
2022-05-07 12:50:10 +08:00
|
|
|
/**
|
|
|
|
* Reads a file and creates a language instance.
|
|
|
|
*/
|
2022-05-11 08:00:47 +08:00
|
|
|
private Language(LanguageStreamDescription description) {
|
2022-05-07 12:50:10 +08:00
|
|
|
@Nullable JsonObject languageData = null;
|
2022-05-11 08:00:47 +08:00
|
|
|
languageCode = description.getLanguageCode();
|
2022-05-10 20:33:45 +08:00
|
|
|
|
|
|
|
try {
|
2022-05-11 08:00:47 +08:00
|
|
|
languageData = Grasscutter.getGsonFactory().fromJson(Utils.readFromInputStream(description.getLanguageFile()), JsonObject.class);
|
2022-05-10 20:33:45 +08:00
|
|
|
} catch (Exception exception) {
|
2022-05-11 08:00:47 +08:00
|
|
|
Grasscutter.getLogger().warn("Failed to load language file: " + description.getLanguageCode(), exception);
|
2022-05-10 20:33:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
this.languageData = languageData;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-05-11 08:00:47 +08:00
|
|
|
* create a LanguageStreamDescription
|
2022-05-10 20:33:45 +08:00
|
|
|
* @param languageCode The name of the language code.
|
|
|
|
* @param fallbackLanguageCode The name of the fallback language code.
|
|
|
|
*/
|
2022-05-11 12:30:07 +08:00
|
|
|
private static LanguageStreamDescription getLanguageFileDescription(String languageCode, String fallbackLanguageCode) {
|
2022-05-10 20:33:45 +08:00
|
|
|
var fileName = languageCode + ".json";
|
|
|
|
var fallback = fallbackLanguageCode + ".json";
|
2022-05-08 06:12:53 +08:00
|
|
|
|
2022-05-10 20:33:45 +08:00
|
|
|
String actualLanguageCode = languageCode;
|
|
|
|
if (cachedLanguages.containsKey(actualLanguageCode)) {
|
2022-05-11 08:00:47 +08:00
|
|
|
return new LanguageStreamDescription(actualLanguageCode, null);
|
2022-05-10 20:33:45 +08:00
|
|
|
}
|
2022-05-11 08:00:47 +08:00
|
|
|
|
2022-05-08 23:53:37 +08:00
|
|
|
InputStream file = Grasscutter.class.getResourceAsStream("/languages/" + fileName);
|
2022-05-10 20:33:45 +08:00
|
|
|
|
2022-05-08 23:58:32 +08:00
|
|
|
if (file == null) { // Provided fallback language.
|
2022-05-11 15:27:05 +08:00
|
|
|
Grasscutter.getLogger().warn("Failed to load language file: " + fileName + ", falling back to: " + fallback);
|
2022-05-10 20:33:45 +08:00
|
|
|
actualLanguageCode = fallbackLanguageCode;
|
|
|
|
if (cachedLanguages.containsKey(actualLanguageCode)) {
|
2022-05-11 08:00:47 +08:00
|
|
|
return new LanguageStreamDescription(actualLanguageCode, null);
|
2022-05-10 20:33:45 +08:00
|
|
|
}
|
2022-05-11 08:00:47 +08:00
|
|
|
|
2022-05-08 23:53:37 +08:00
|
|
|
file = Grasscutter.class.getResourceAsStream("/languages/" + fallback);
|
2022-05-08 23:58:32 +08:00
|
|
|
}
|
2022-05-10 20:33:45 +08:00
|
|
|
|
2022-05-08 23:58:32 +08:00
|
|
|
if(file == null) { // Fallback the fallback language.
|
2022-05-11 15:27:05 +08:00
|
|
|
Grasscutter.getLogger().warn("Failed to load language file: " + fallback + ", falling back to: en-US.json");
|
2022-05-10 20:33:45 +08:00
|
|
|
actualLanguageCode = "en-US";
|
|
|
|
if (cachedLanguages.containsKey(actualLanguageCode)) {
|
2022-05-11 08:00:47 +08:00
|
|
|
return new LanguageStreamDescription(actualLanguageCode, null);
|
2022-05-10 20:33:45 +08:00
|
|
|
}
|
2022-05-11 08:00:47 +08:00
|
|
|
|
2022-05-08 23:53:37 +08:00
|
|
|
file = Grasscutter.class.getResourceAsStream("/languages/en-US.json");
|
2022-05-08 23:58:32 +08:00
|
|
|
}
|
2022-05-10 20:33:45 +08:00
|
|
|
|
2022-05-08 23:53:37 +08:00
|
|
|
if(file == null)
|
|
|
|
throw new RuntimeException("Unable to load the primary, fallback, and 'en-US' language files.");
|
2022-05-10 20:33:45 +08:00
|
|
|
|
2022-05-11 08:00:47 +08:00
|
|
|
return new LanguageStreamDescription(actualLanguageCode, file);
|
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;
|
|
|
|
}
|
2022-05-10 20:33:45 +08:00
|
|
|
|
2022-05-11 08:00:47 +08:00
|
|
|
private static class LanguageStreamDescription {
|
|
|
|
private final String languageCode;
|
|
|
|
private final InputStream languageFile;
|
2022-05-10 20:33:45 +08:00
|
|
|
|
2022-05-11 08:00:47 +08:00
|
|
|
public LanguageStreamDescription(String languageCode, InputStream languageFile) {
|
2022-05-10 20:33:45 +08:00
|
|
|
this.languageCode = languageCode;
|
|
|
|
this.languageFile = languageFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getLanguageCode() {
|
|
|
|
return languageCode;
|
|
|
|
}
|
|
|
|
|
|
|
|
public InputStream getLanguageFile() {
|
|
|
|
return languageFile;
|
|
|
|
}
|
|
|
|
}
|
2022-05-06 11:57:55 +08:00
|
|
|
}
|