locales-coverage-description.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. const fs = require("fs");
  2. const THRESSHOLD = 85;
  3. const crowdinMap = {
  4. "ar-SA": "en-ar",
  5. "bg-BG": "en-bg",
  6. "ca-ES": "en-ca",
  7. "de-DE": "en-de",
  8. "el-GR": "en-el",
  9. "es-ES": "en-es",
  10. "fa-IR": "en-fa",
  11. "fi-FI": "en-fi",
  12. "fr-FR": "en-fr",
  13. "he-IL": "en-he",
  14. "hi-IN": "en-hi",
  15. "hu-HU": "en-hu",
  16. "id-ID": "en-id",
  17. "it-IT": "en-it",
  18. "ja-JP": "en-ja",
  19. "kab-KAB": "en-kab",
  20. "ko-KR": "en-ko",
  21. "my-MM": "en-my",
  22. "nb-NO": "en-nb",
  23. "nl-NL": "en-nl",
  24. "nn-NO": "en-nnno",
  25. "oc-FR": "en-oc",
  26. "pa-IN": "en-pain",
  27. "pl-PL": "en-pl",
  28. "pt-BR": "en-ptbr",
  29. "pt-PT": "en-pt",
  30. "ro-RO": "en-ro",
  31. "ru-RU": "en-ru",
  32. "sk-SK": "en-sk",
  33. "sv-SE": "en-sv",
  34. "tr-TR": "en-tr",
  35. "uk-UA": "en-uk",
  36. "zh-CN": "en-zhcn",
  37. "zh-TW": "en-zhtw",
  38. "lv-LV": "en-lv",
  39. "cs-CZ": "cs-cz",
  40. };
  41. const flags = {
  42. "ar-SA": "🇸🇦",
  43. "bg-BG": "🇧🇬",
  44. "ca-ES": "🏳",
  45. "de-DE": "🇩🇪",
  46. "el-GR": "🇬🇷",
  47. "es-ES": "🇪🇸",
  48. "fa-IR": "🇮🇷",
  49. "fi-FI": "🇫🇮",
  50. "fr-FR": "🇫🇷",
  51. "he-IL": "🇮🇱",
  52. "hi-IN": "🇮🇳",
  53. "hu-HU": "🇭🇺",
  54. "id-ID": "🇮🇩",
  55. "it-IT": "🇮🇹",
  56. "ja-JP": "🇯🇵",
  57. "kab-KAB": "🏳",
  58. "ko-KR": "🇰🇷",
  59. "my-MM": "🇲🇲",
  60. "nb-NO": "🇳🇴",
  61. "nl-NL": "🇳🇱",
  62. "nn-NO": "🇳🇴",
  63. "oc-FR": "🏳",
  64. "pa-IN": "🇮🇳",
  65. "pl-PL": "🇵🇱",
  66. "pt-BR": "🇧🇷",
  67. "pt-PT": "🇵🇹",
  68. "ro-RO": "🇷🇴",
  69. "ru-RU": "🇷🇺",
  70. "sk-SK": "🇸🇰",
  71. "sv-SE": "🇸🇪",
  72. "tr-TR": "🇹🇷",
  73. "uk-UA": "🇺🇦",
  74. "zh-CN": "🇨🇳",
  75. "zh-TW": "🇹🇼",
  76. "lv-LV": "🇱🇻",
  77. "cs-CZ": "🇨🇿",
  78. };
  79. const languages = {
  80. "ar-SA": "العربية",
  81. "bg-BG": "Български",
  82. "ca-ES": "Català",
  83. "de-DE": "Deutsch",
  84. "el-GR": "Ελληνικά",
  85. "es-ES": "Español",
  86. "fa-IR": "فارسی",
  87. "fi-FI": "Suomi",
  88. "fr-FR": "Français",
  89. "he-IL": "עברית",
  90. "hi-IN": "हिन्दी",
  91. "hu-HU": "Magyar",
  92. "id-ID": "Bahasa Indonesia",
  93. "it-IT": "Italiano",
  94. "ja-JP": "日本語",
  95. "kab-KAB": "Taqbaylit",
  96. "ko-KR": "한국어",
  97. "my-MM": "Burmese",
  98. "nb-NO": "Norsk bokmål",
  99. "nl-NL": "Nederlands",
  100. "nn-NO": "Norsk nynorsk",
  101. "oc-FR": "Occitan",
  102. "pa-IN": "ਪੰਜਾਬੀ",
  103. "pl-PL": "Polski",
  104. "pt-BR": "Português Brasileiro",
  105. "pt-PT": "Português",
  106. "ro-RO": "Română",
  107. "ru-RU": "Русский",
  108. "sk-SK": "Slovenčina",
  109. "sv-SE": "Svenska",
  110. "tr-TR": "Türkçe",
  111. "uk-UA": "Українська",
  112. "zh-CN": "简体中文",
  113. "zh-TW": "繁體中文",
  114. "lv-LV": "Latviešu",
  115. "cs-CZ": "Česky",
  116. };
  117. const percentages = fs.readFileSync(
  118. `${__dirname}/../src/locales/percentages.json`,
  119. );
  120. const rowData = JSON.parse(percentages);
  121. const coverages = Object.entries(rowData)
  122. .sort(([, a], [, b]) => b - a)
  123. .reduce((r, [k, v]) => ({ ...r, [k]: v }), {});
  124. const boldIf = (text, condition) => (condition ? `**${text}**` : text);
  125. const printHeader = () => {
  126. let result = "| | Flag | Locale | % |\n";
  127. result += "| :--: | :--: | -- | :--: |";
  128. return result;
  129. };
  130. const printRow = (id, locale, coverage) => {
  131. const isOver = coverage >= THRESSHOLD;
  132. let result = `| ${isOver ? id : "..."} | `;
  133. result += `${locale in flags ? flags[locale] : ""} | `;
  134. const language = locale in languages ? languages[locale] : locale;
  135. if (locale in crowdinMap && crowdinMap[locale]) {
  136. result += `[${boldIf(
  137. language,
  138. isOver,
  139. )}](https://crowdin.com/translate/excalidraw/10/${crowdinMap[locale]}) | `;
  140. } else {
  141. result += `${boldIf(language, isOver)} | `;
  142. }
  143. result += `${coverage === 100 ? "💯" : boldIf(coverage, isOver)} |`;
  144. return result;
  145. };
  146. console.info(
  147. `Each language must be at least **${THRESSHOLD}%** translated in order to appear on Excalidraw. Join us on [Crowdin](https://crowdin.com/project/excalidraw) and help us translate your own language. **Can't find yours yet?** Open an [issue](https://github.com/excalidraw/excalidraw/issues/new) and we'll add it to the list.`,
  148. );
  149. console.info("\n\r");
  150. console.info(printHeader());
  151. let index = 1;
  152. for (const coverage in coverages) {
  153. if (coverage === "en") {
  154. continue;
  155. }
  156. console.info(printRow(index, coverage, coverages[coverage]));
  157. index++;
  158. }
  159. console.info("\n\r");
  160. console.info("\\* Languages in **bold** are going to appear on production.");