Languages.h 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. /*
  2. * Languages.h, part of VCMI engine
  3. *
  4. * Authors: listed in file AUTHORS in main folder
  5. *
  6. * License: GNU General Public License v2.0 or later
  7. * Full text of license available in license.txt file, in main folder
  8. *
  9. */
  10. #pragma once
  11. namespace Languages
  12. {
  13. enum class EPluralForms
  14. {
  15. NONE,
  16. VI_1, // Single plural form, (Vietnamese)
  17. EN_2, // Two forms, singular used for one only (English)
  18. FR_2, // Two forms, singular used for zero and one (French)
  19. UK_3, // Three forms, special cases for numbers ending in 1 and 2, 3, 4, except those ending in 1[1-4] (Ukrainian)
  20. CZ_3, // Three forms, special cases for 1 and 2, 3, 4 (Czech)
  21. PL_3, // Three forms, special case for one and some numbers ending in 2, 3, or 4 (Polish)
  22. };
  23. enum class ELanguages
  24. {
  25. CZECH,
  26. CHINESE,
  27. ENGLISH,
  28. FINNISH,
  29. FRENCH,
  30. GERMAN,
  31. HUNGARIAN,
  32. ITALIAN,
  33. KOREAN,
  34. POLISH,
  35. PORTUGUESE,
  36. RUSSIAN,
  37. SPANISH,
  38. SWEDISH,
  39. TURKISH,
  40. UKRAINIAN,
  41. VIETNAMESE,
  42. // Pseudo-languages, that have no translations but can define H3 encoding to use
  43. OTHER_CP1250,
  44. OTHER_CP1251,
  45. OTHER_CP1252,
  46. COUNT
  47. };
  48. struct Options
  49. {
  50. /// string identifier (ascii, lower-case), e.g. "english"
  51. std::string identifier;
  52. /// human-readable name of language in English
  53. std::string nameEnglish;
  54. /// human-readable name of language in its own language
  55. std::string nameNative;
  56. /// encoding that is used by H3 for this language
  57. std::string encoding;
  58. /// primary IETF language tag
  59. std::string tagIETF;
  60. /// DateTime format
  61. std::string dateTimeFormat;
  62. /// Ruleset for plural forms in this language
  63. EPluralForms pluralForms = EPluralForms::NONE;
  64. /// VCMI supports translations into this language
  65. bool hasTranslation = false;
  66. };
  67. inline const auto & getLanguageList()
  68. {
  69. static const std::array<Options, 20> languages
  70. { {
  71. { "czech", "Czech", "Čeština", "CP1250", "cs", "%d.%m.%Y %T", EPluralForms::CZ_3, true },
  72. { "chinese", "Chinese", "简体中文", "GBK", "zh", "%F %T", EPluralForms::VI_1, true }, // Note: actually Simplified Chinese
  73. { "english", "English", "English", "CP1252", "en", "%F %T", EPluralForms::EN_2, true }, // English uses international date/time format here
  74. { "finnish", "Finnish", "Suomi", "CP1252", "fi", "%d.%m.%Y %T", EPluralForms::EN_2, true },
  75. { "french", "French", "Français", "CP1252", "fr", "%d/%m/%Y %T", EPluralForms::FR_2, true },
  76. { "german", "German", "Deutsch", "CP1252", "de", "%d.%m.%Y %T", EPluralForms::EN_2, true },
  77. { "hungarian", "Hungarian", "Magyar", "CP1250", "hu", "%Y. %m. %d. %T", EPluralForms::EN_2, true },
  78. { "italian", "Italian", "Italiano", "CP1250", "it", "%d/%m/%Y %T", EPluralForms::EN_2, true },
  79. { "korean", "Korean", "한국어", "CP949", "ko", "%F %T", EPluralForms::VI_1, true },
  80. { "polish", "Polish", "Polski", "CP1250", "pl", "%d.%m.%Y %T", EPluralForms::PL_3, true },
  81. { "portuguese", "Portuguese", "Português", "CP1252", "pt", "%d/%m/%Y %T", EPluralForms::EN_2, true }, // Note: actually Brazilian Portuguese
  82. { "russian", "Russian", "Русский", "CP1251", "ru", "%d.%m.%Y %T", EPluralForms::UK_3, true },
  83. { "spanish", "Spanish", "Español", "CP1252", "es", "%d/%m/%Y %T", EPluralForms::EN_2, true },
  84. { "swedish", "Swedish", "Svenska", "CP1252", "sv", "%F %T", EPluralForms::EN_2, true },
  85. { "turkish", "Turkish", "Türkçe", "CP1254", "tr", "%d.%m.%Y %T", EPluralForms::EN_2, true },
  86. { "ukrainian", "Ukrainian", "Українська", "CP1251", "uk", "%d.%m.%Y %T", EPluralForms::UK_3, true },
  87. { "vietnamese", "Vietnamese", "Tiếng Việt", "UTF-8", "vi", "%d/%m/%Y %T", EPluralForms::VI_1, true }, // Fan translation uses special encoding
  88. { "other_cp1250", "Other (East European)", "", "CP1250", "", "", EPluralForms::NONE, false },
  89. { "other_cp1251", "Other (Cyrillic Script)", "", "CP1251", "", "", EPluralForms::NONE, false },
  90. { "other_cp1252", "Other (West European)", "", "CP1252", "", "", EPluralForms::NONE, false }
  91. } };
  92. static_assert(languages.size() == static_cast<size_t>(ELanguages::COUNT), "Languages array is missing a value!");
  93. return languages;
  94. }
  95. inline const Options & getLanguageOptions(ELanguages language)
  96. {
  97. return getLanguageList().at(static_cast<size_t>(language));
  98. }
  99. inline const Options & getLanguageOptions(const std::string & language)
  100. {
  101. for(const auto & entry : getLanguageList())
  102. if(entry.identifier == language)
  103. return entry;
  104. throw std::out_of_range("Language " + language + " does not exists!");
  105. }
  106. template<typename Numeric>
  107. inline constexpr int getPluralFormIndex(EPluralForms form, Numeric value)
  108. {
  109. // Based on https://www.gnu.org/software/gettext/manual/html_node/Plural-forms.html
  110. switch(form)
  111. {
  112. case EPluralForms::NONE:
  113. case EPluralForms::VI_1:
  114. return 0;
  115. case EPluralForms::EN_2:
  116. if (value == 1)
  117. return 1;
  118. return 2;
  119. case EPluralForms::FR_2:
  120. if (value == 1 || value == 0)
  121. return 1;
  122. return 2;
  123. case EPluralForms::UK_3:
  124. if (value % 10 == 1 && value % 100 != 11)
  125. return 1;
  126. if (value%10>=2 && value%10<=4 && (value%100<10 || value%100>=20))
  127. return 2;
  128. return 0;
  129. case EPluralForms::CZ_3:
  130. if (value == 1)
  131. return 1;
  132. if (value>=2 && value<=4)
  133. return 2;
  134. return 0;
  135. case EPluralForms::PL_3:
  136. if (value == 1)
  137. return 1;
  138. if (value%10>=2 && value%10<=4 && (value%100<10 || value%100>=20))
  139. return 2;
  140. return 0;
  141. }
  142. throw std::runtime_error("Invalid plural form enumeration received!");
  143. }
  144. template<typename Numeric>
  145. inline std::string getPluralFormTextID(std::string languageName, Numeric value, std::string textID)
  146. {
  147. int formIndex = getPluralFormIndex(getLanguageOptions(languageName).pluralForms, value);
  148. return textID + '.' + std::to_string(formIndex);
  149. }
  150. }