locales-coverage-description.js 4.3 KB

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