package org.multibit.hd.ui.languages; import org.multibit.hd.ui.views.components.Images; import javax.swing.*; import java.awt.*; import java.util.Locale; import java.util.ResourceBundle; import static org.multibit.hd.ui.views.fonts.TitleFontDecorator.*; /** * <p>Enum to provide the following to application:</p> * <ul> * <li>Language keys to use for supported locales (language,region,variant)</li> * </ul> * * <h3>Naming conventions</h3> * <p>Language keys are placed in an enum for type safety.</p> * <p>Language keys have their resource key so that IDEs can maintain a "where used" reference lookup.</p> * <p>Resource keys are simply the language key enum name in lower case and are only present in the * base resource bundle (e.g. "viewer.properties").</p> * <p>Eac languages has a preferred title font for presentation</p> * * @since 0.0.1 * */ public enum LanguageKey { /** * Afrikaans in Africa */ AF_AF("af_AF", CORBEN_REGULAR), /** * Arabic */ AR_AR("ar_AR", IMPACT_REGULAR), /** * Bulgarian */ BG_BG("bg_BG", OPENSANS_SEMIBOLD), /** * Czech */ CS_CZ("cs_CZ", OPENSANS_SEMIBOLD), /** * Catalan */ CA_ES("ca_ES", CORBEN_REGULAR), /** * Danish */ DA_DK("da_DK", CORBEN_REGULAR), /** * German */ DE_DE("de_DE", CORBEN_REGULAR), /** * Greek */ EL_GR("el_GR", OPENSANS_SEMIBOLD), /** * English (United Kingdom) */ EN_GB("en_GB", CORBEN_REGULAR), /** * English (United States) */ EN_US("en_US", CORBEN_REGULAR), /** * Esperanto */ EO("eo", CORBEN_REGULAR), // Esperanto has no country /** * Spanish */ ES_ES("es_ES", CORBEN_REGULAR), /** * Farsi */ FA_IR("fa_IR", IMPACT_REGULAR), /** * Finnish */ FI_FI("fi_FI", CORBEN_REGULAR), /** * French */ FR_FR("fr_FR", CORBEN_REGULAR), /** * Hindi (should use NotoSans-Bold when proven) */ HI_IN("hi_IN", IMPACT_REGULAR), /** * Croatian */ HR_HR("hr_HR", OPENSANS_SEMIBOLD), /** * Hungarian */ HU_HU("hu_HU", OPENSANS_SEMIBOLD), /** * Indonesian */ IN_ID("in_ID", IMPACT_REGULAR), // Legacy form of "id_ID" for Indonesian in Indonesia /** * Hebrew (Israel) */ IW_IL("iw_IL", IMPACT_REGULAR), // Legacy form of "he_IL" for Hebrew in Israel /** * Italian */ IT_IT("it_IT", CORBEN_REGULAR), /** * Japanese */ JA_JP("ja_JP", IMPACT_REGULAR), /** * Korean */ KO_KR("ko_KR", IMPACT_REGULAR), /** * Latvian */ LV_LV("lv_LV", OPENSANS_SEMIBOLD), /** * Lithuanian */ LT_LT("lt_LT", OPENSANS_SEMIBOLD), /** * Mongolian */ MN_MN("mn_MN", IMPACT_REGULAR), /** * Dutch */ NL_NL("nl_NL", CORBEN_REGULAR), /** * Norwegian */ NO_NO("no_NO", CORBEN_REGULAR), /** * Polish */ PL_PL("pl_PL", OPENSANS_SEMIBOLD), /** * Portuguese (Brazil) */ PT_BR("pt_BR", CORBEN_REGULAR), /** * Portuguese (Portugal) */ PT_PT("pt_PT", CORBEN_REGULAR), /** * Romanian */ RO_RO("ro_RO", OPENSANS_SEMIBOLD), /** * Russian */ RU_RU("ru_RU", OPENSANS_SEMIBOLD), /** * Serbian (Latin) */ SR_CS("sr_CS", OPENSANS_SEMIBOLD), /** * Slovak */ SK_SK("sk_SK", OPENSANS_SEMIBOLD), /** * Slovene */ SL_SI("sl_SI", OPENSANS_SEMIBOLD), /** * Swedish */ SV_SV("sv_SV", CORBEN_REGULAR), /** * Swahili */ SW_KE("sw_KE", IMPACT_REGULAR), /** * Tamil */ TA_LK("ta_LK", IMPACT_REGULAR), /** * Thai */ TH_TH("th_TH", IMPACT_REGULAR), /** * Tagalog */ TL_PH("tl_PH", IMPACT_REGULAR), /** * Turkish */ TR_TR("tr_TR", IMPACT_REGULAR), /** * Vietnamese */ VI_VN("vi_VN", OPENSANS_SEMIBOLD), /** * Chinese (Simplified) */ ZH_CN("zh_CN", IMPACT_REGULAR), /** * Chinese (Traditional) */ ZH_TW("zh_TW", IMPACT_REGULAR), // End of enum ; private final String key; private final String countryCode; private final String languageCode; private final String languageName; private final Font titleFont; private ImageIcon icon; private LanguageKey(String key, Font titleFont) { this.titleFont = titleFont; // The bundle is cached by the JVM ResourceBundle rb = ResourceBundle.getBundle(Languages.BASE_NAME); this.key = key; this.languageCode = key.substring(0, 2); if (key.contains("_")) { this.countryCode = key.substring(3, 5); } else { this.countryCode = languageCode; } this.icon = Images.newLanguageCodeIcon(languageCode); this.languageName = rb.getString(key); } /** * <p>Reset the icons after a theme switch</p> */ public static void resetIcons() { for (LanguageKey languageKey : values()) { languageKey.icon = Images.newLanguageCodeIcon(languageKey.languageCode); } } /** * @return The appropriate title font */ public Font getTitleFont() { return titleFont; } /** * @return The key for use with the resource bundles */ public String getKey() { return key; } /** * @return The icon for use with a list renderer */ public ImageIcon getIcon() { return icon; } /** * @return The 2-letter country code (e.g. "UK") */ public String getCountryCode() { return countryCode; } /** * @return The 2-letter language code (e.g. "en") */ public String getLanguageCode() { return languageCode; } /** * @return The language name (e.g. "English (United Kingdom)") */ public String getLanguageName() { return languageName; } /** * @param locale The locale providing at least language and region * * @return The matching language key, or the default EN_US since it is dominant on the internet */ public static LanguageKey fromLocale(Locale locale) { // Ensure we use English rules for uppercase to identify enum keys // We use the legacy names for countries for consistency String language = locale.getLanguage().toUpperCase(Locale.ENGLISH); String country = locale.getCountry().toUpperCase(Locale.ENGLISH); String variant = locale.getVariant().toUpperCase(Locale.ENGLISH); String matcher1 = language + "_" + country + "_" + variant; String matcher2 = language + "_" + country; for (LanguageKey languageKey : values()) { // Language, country and variant if (languageKey.name().equals(matcher1)) { return languageKey; } // Language and country if (languageKey.name().equals(matcher2)) { return languageKey; } // At this point we match only on language (e.g. "EO" for Esperanto) // so that we don't introduce a country or region bias // Language only if (languageKey.name().equals(matcher2)) { return languageKey; } } // We have an unsupported locale so we use the first entry that matches // the supported language // Find the first entry with the supported language for (LanguageKey languageKey : values()) { if (languageKey.name().substring(0, 2).equals(language)) { return languageKey; } } // Unsupported language so default to EN_US since it is the dominant locale on the internet return LanguageKey.EN_US; } /** * @param languageName The language name (e.g. "English (United Kingdom)") as specified in the primary resource bundle * * @return The language key matching the language name */ public static LanguageKey fromLanguageName(String languageName) { // Use the resource bundle translations for (LanguageKey languageKey : values()) { if (languageKey.getLanguageName().equalsIgnoreCase(languageName)) { return languageKey; } } // Unknown language name so fall back to Java locale lookup for current locale for (Locale locale : Locale.getAvailableLocales()) { if (locale.getDisplayName().equalsIgnoreCase(languageName)) { return fromLocale(locale); } } throw new IllegalArgumentException("'languageName' was not matched for '" + languageName + "'"); } /** * @return The localised names of the languages in the order they are declared */ public static String[] localisedNames() { String[] languageNames = new String[values().length]; int i = 0; for (LanguageKey languageKey : values()) { languageNames[i] = languageKey.getLanguageName(); i++; } return languageNames; } }