Converse converse.js

Source: i18n/index.js

  1. /**
  2. * @module i18n
  3. * @copyright 2022, the Converse.js contributors
  4. * @license Mozilla Public License (MPLv2)
  5. * @description This is the internationalization module
  6. */
  7. import Jed from 'jed';
  8. import log from '@converse/headless/log.js';
  9. import { _converse, api, converse, i18n } from '@converse/headless/core.js';
  10. const { dayjs } = converse.env;
  11. let jed_instance;
  12. /**
  13. * @private
  14. * @param { string } locale
  15. * @param { string[] } supported_locales
  16. */
  17. function isConverseLocale (locale, supported_locales) {
  18. return typeof locale === 'string' && supported_locales.includes(locale);
  19. }
  20. /**
  21. * Determines which locale is supported by the user's system as well
  22. * as by the relevant library (e.g. converse.js or dayjs).
  23. * @private
  24. * @param { string } preferred_locale
  25. * @param { Function } isSupportedByLibrary - Returns a boolean indicating whether
  26. * the locale is supported.
  27. * @returns { string }
  28. */
  29. function getLocale (preferred_locale, isSupportedByLibrary) {
  30. if (preferred_locale === 'en' || isSupportedByLibrary(preferred_locale)) {
  31. return preferred_locale;
  32. }
  33. const { languages } = window.navigator;
  34. let locale;
  35. for (let i = 0; i < languages.length && !locale; i++) {
  36. locale = isLocaleAvailable(languages[i], isSupportedByLibrary);
  37. }
  38. return locale || 'en';
  39. }
  40. /**
  41. * Check whether the locale or sub locale (e.g. en-US, en) is supported.
  42. * @private
  43. * @param { String } locale - The locale to check for
  44. * @param { Function } available - Returns a boolean indicating whether the locale is supported
  45. */
  46. function isLocaleAvailable (locale, available) {
  47. if (available(locale)) {
  48. return locale;
  49. } else {
  50. var sublocale = locale.split('-')[0];
  51. if (sublocale !== locale && available(sublocale)) {
  52. return sublocale;
  53. }
  54. }
  55. }
  56. /**
  57. * Given a locale, return the closest locale returned by dayJS
  58. * @private
  59. * @param { string } locale
  60. */
  61. function getDayJSLocale (locale) {
  62. const dayjs_locale = locale.toLowerCase().replace('_', '-');
  63. return dayjs_locale === 'ug' ? 'ug-cn' : dayjs_locale;
  64. }
  65. /**
  66. * Fetch the translations for the given local at the given URL.
  67. * @private
  68. * @returns { Jed }
  69. */
  70. async function fetchTranslations () {
  71. const { api, locale } = _converse;
  72. const dayjs_locale = getDayJSLocale(locale);
  73. if (!isConverseLocale(locale, api.settings.get('locales')) || locale === 'en') {
  74. return;
  75. }
  76. const { default: data } = await import(
  77. /*webpackChunkName: "locales/[request]" */ `../i18n/${locale}/LC_MESSAGES/converse.po`
  78. );
  79. await import(/*webpackChunkName: "locales/dayjs/[request]" */ `dayjs/locale/${dayjs_locale}.js`);
  80. dayjs.locale(getLocale(dayjs_locale, (l) => dayjs.locale(l)));
  81. return new Jed(data);
  82. }
  83. /**
  84. * @namespace i18n
  85. */
  86. Object.assign(i18n, {
  87. /**
  88. * @param { string } preferred_locale
  89. * @param { string[] } available_locales
  90. */
  91. getLocale (preferred_locale, available_locales) {
  92. return getLocale(preferred_locale, (preferred) => isConverseLocale(preferred, available_locales));
  93. },
  94. /**
  95. * @param { string } str - The string to be translated
  96. */
  97. translate (str) {
  98. if (!jed_instance) {
  99. return Jed.sprintf.apply(Jed, arguments);
  100. }
  101. const t = jed_instance.translate(str);
  102. if (arguments.length > 1) {
  103. return t.fetch.apply(t, [].slice.call(arguments, 1));
  104. } else {
  105. return t.fetch();
  106. }
  107. },
  108. async initialize () {
  109. if (_converse.isTestEnv()) {
  110. _converse.locale = 'en';
  111. } else {
  112. try {
  113. const preferred_locale = api.settings.get('i18n');
  114. _converse.locale = i18n.getLocale(preferred_locale, api.settings.get('locales'));
  115. jed_instance = await fetchTranslations();
  116. } catch (e) {
  117. log.fatal(e.message);
  118. _converse.locale = 'en';
  119. }
  120. }
  121. },
  122. __ (...args) {
  123. return i18n.translate(...args);
  124. },
  125. });
  126. export const __ = i18n.__;