You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

locales-coverage-description.js 4.2KB

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