Source: ui/localization.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.ui.Localization');
  7. goog.provide('shaka.ui.Localization.ConflictResolution');
  8. goog.require('shaka.util.FakeEvent');
  9. goog.require('shaka.util.FakeEventTarget');
  10. goog.require('shaka.util.Iterables');
  11. goog.require('shaka.util.LanguageUtils');
  12. // TODO: link to the design and usage documentation here
  13. // b/117679670
  14. /**
  15. * Localization system provided by the shaka ui library.
  16. * It can be used to store the various localized forms of
  17. * strings that are expected to be displayed to the user.
  18. * If a string is not available, it will return the localized
  19. * form in the closest related locale.
  20. *
  21. * @final
  22. * @export
  23. */
  24. shaka.ui.Localization = class extends shaka.util.FakeEventTarget {
  25. /**
  26. * @param {string} fallbackLocale
  27. * The fallback locale that should be used. It will be assumed that this
  28. * locale should have entries for just about every request.
  29. */
  30. constructor(fallbackLocale) {
  31. super();
  32. /** @private {string} */
  33. this.fallbackLocale_ = shaka.util.LanguageUtils.normalize(fallbackLocale);
  34. /**
  35. * The current mappings that will be used when requests are made. Since
  36. * nothing has been loaded yet, there will be nothing in this map.
  37. *
  38. * @private {!Map.<string, string>}
  39. */
  40. this.currentMap_ = new Map();
  41. /**
  42. * The locales that were used when creating |currentMap_|. Since we don't
  43. * have anything when we first initialize, an empty set means "no
  44. * preference".
  45. *
  46. * @private {!Set.<string>}
  47. */
  48. this.currentLocales_ = new Set();
  49. /**
  50. * A map of maps where:
  51. * - The outer map is a mapping from locale code to localizations.
  52. * - The inner map is a mapping from id to localized text.
  53. *
  54. * @private {!Map.<string, !Map.<string, string>>}
  55. */
  56. this.localizations_ = new Map();
  57. }
  58. /**
  59. * @override
  60. * @export
  61. */
  62. release() {
  63. // Placeholder so that readers know this implements IReleasable (via
  64. // FakeEventTarget)
  65. super.release();
  66. }
  67. /**
  68. * Request the localization system to change which locale it serves. If any of
  69. * of the preferred locales cannot be found, the localization system will fire
  70. * an event identifying which locales it does not know. The localization
  71. * system will then continue to operate using the closest matches it has.
  72. *
  73. * @param {!Iterable.<string>} locales
  74. * The locale codes for the requested locales in order of preference.
  75. * @export
  76. */
  77. changeLocale(locales) {
  78. const Class = shaka.ui.Localization;
  79. // Normalize the locale so that matching will be easier. We need to reset
  80. // our internal set of locales so that we have the same order as the new
  81. // set.
  82. this.currentLocales_.clear();
  83. for (const locale of locales) {
  84. this.currentLocales_.add(shaka.util.LanguageUtils.normalize(locale));
  85. }
  86. this.updateCurrentMap_();
  87. // Check if we have support for the exact locale requested. Even through we
  88. // will do our best to return the most relevant results, we need to tell
  89. // app that some data may be missing.
  90. const missing = shaka.util.Iterables.filter(
  91. this.currentLocales_,
  92. (locale) => !this.localizations_.has(locale));
  93. if (missing.length) {
  94. this.dispatchEvent(new shaka.util.FakeEvent(
  95. Class.UNKNOWN_LOCALES,
  96. (new Map()).set('locales', missing)));
  97. }
  98. const found = shaka.util.Iterables.filter(
  99. this.currentLocales_,
  100. (locale) => this.localizations_.has(locale));
  101. const data = (new Map()).set(
  102. 'locales', found.length ? found : [this.fallbackLocale_]);
  103. this.dispatchEvent(new shaka.util.FakeEvent(
  104. Class.LOCALE_CHANGED,
  105. data));
  106. }
  107. /**
  108. * Insert a set of localizations for a single locale. This will amend the
  109. * existing localizations for the given locale.
  110. *
  111. * @param {string} locale
  112. * The locale that the localizations should be added to.
  113. * @param {!Map.<string, string>} localizations
  114. * A mapping of id to localized text that should used to modify the internal
  115. * collection of localizations.
  116. * @param {shaka.ui.Localization.ConflictResolution=} conflictResolution
  117. * The strategy used to resolve conflicts when the id of an existing entry
  118. * matches the id of a new entry. Default to |USE_NEW|, where the new
  119. * entry will replace the old entry.
  120. * @return {!shaka.ui.Localization}
  121. * Returns |this| so that calls can be chained.
  122. * @export
  123. */
  124. insert(locale, localizations, conflictResolution) {
  125. const Class = shaka.ui.Localization;
  126. const ConflictResolution = shaka.ui.Localization.ConflictResolution;
  127. const FakeEvent = shaka.util.FakeEvent;
  128. // Normalize the locale so that matching will be easier.
  129. locale = shaka.util.LanguageUtils.normalize(locale);
  130. // Default |conflictResolution| to |USE_NEW| if it was not given. Doing it
  131. // here because it would create too long of a parameter list.
  132. if (conflictResolution === undefined) {
  133. conflictResolution = ConflictResolution.USE_NEW;
  134. }
  135. // Make sure we have an entry for the locale because we are about to
  136. // write to it.
  137. const table = this.localizations_.get(locale) || new Map();
  138. localizations.forEach((value, id) => {
  139. // Set the value if we don't have an old value or if we are to replace
  140. // the old value with the new value.
  141. if (!table.has(id) || conflictResolution == ConflictResolution.USE_NEW) {
  142. table.set(id, value);
  143. }
  144. });
  145. this.localizations_.set(locale, table);
  146. // The data we use to make our map may have changed, update the map we pull
  147. // data from.
  148. this.updateCurrentMap_();
  149. this.dispatchEvent(new FakeEvent(Class.LOCALE_UPDATED));
  150. return this;
  151. }
  152. /**
  153. * Set the value under each key in |dictionary| to the resolved value.
  154. * Convenient for apps with some kind of data binding system.
  155. *
  156. * Equivalent to:
  157. * for (const key of dictionary.keys()) {
  158. * dictionary.set(key, localization.resolve(key));
  159. * }
  160. *
  161. * @param {!Map.<string, string>} dictionary
  162. * @export
  163. */
  164. resolveDictionary(dictionary) {
  165. for (const key of dictionary.keys()) {
  166. // Since we are not changing what keys are in the map, it is safe to
  167. // update the map while iterating it.
  168. dictionary.set(key, this.resolve(key));
  169. }
  170. }
  171. /**
  172. * Request the localized string under the given id. If there is no localized
  173. * version of the string, then the fallback localization will be given
  174. * ("en" version). If there is no fallback localization, a non-null empty
  175. * string will be returned.
  176. *
  177. * @param {string} id The id for the localization entry.
  178. * @return {string}
  179. * @export
  180. */
  181. resolve(id) {
  182. const Class = shaka.ui.Localization;
  183. const FakeEvent = shaka.util.FakeEvent;
  184. /** @type {string} */
  185. const result = this.currentMap_.get(id);
  186. // If we have a result, it means that it was found in either the current
  187. // locale or one of the fall-backs.
  188. if (result) {
  189. return result;
  190. }
  191. // Since we could not find the result, it means it is missing from a large
  192. // number of locales. Since we don't know which ones we actually checked,
  193. // just tell them the preferred locale.
  194. const data = new Map()
  195. // Make a copy to avoid leaking references.
  196. .set('locales', Array.from(this.currentLocales_))
  197. .set('missing', id);
  198. this.dispatchEvent(new FakeEvent(Class.UNKNOWN_LOCALIZATION, data));
  199. return '';
  200. }
  201. /**
  202. * The locales currently used. An empty set means "no preference".
  203. *
  204. * @return {!Set.<string>}
  205. * @export
  206. */
  207. getCurrentLocales() {
  208. return this.currentLocales_;
  209. }
  210. /**
  211. * @private
  212. */
  213. updateCurrentMap_() {
  214. const LanguageUtils = shaka.util.LanguageUtils;
  215. /** @type {!Map.<string, !Map.<string, string>>} */
  216. const localizations = this.localizations_;
  217. /** @type {string} */
  218. const fallbackLocale = this.fallbackLocale_;
  219. /** @type {!Iterable.<string>} */
  220. const preferredLocales = this.currentLocales_;
  221. /**
  222. * We want to create a single map that gives us the best possible responses
  223. * for the current locale. To do this, we will go through be loosest
  224. * matching locales to the best matching locales. By the time we finish
  225. * flattening the maps, the best result will be left under each key.
  226. *
  227. * Get the locales we should use in order of preference. For example with
  228. * preferred locales of "elvish-WOODLAND" and "dwarfish-MOUNTAIN" and a
  229. * fallback of "common-HUMAN", this would look like:
  230. *
  231. * new Set([
  232. * // Preference 1
  233. * 'elvish-WOODLAND',
  234. * // Preference 1 Base
  235. * 'elvish',
  236. * // Preference 1 Siblings
  237. * 'elvish-WOODLAND', 'elvish-WESTWOOD', 'elvish-MARSH,
  238. * // Preference 2
  239. * 'dwarfish-MOUNTAIN',
  240. * // Preference 2 base
  241. * 'dwarfish',
  242. * // Preference 2 Siblings
  243. * 'dwarfish-MOUNTAIN', 'dwarfish-NORTH', "dwarish-SOUTH",
  244. * // Fallback
  245. * 'common-HUMAN',
  246. * ])
  247. *
  248. * @type {!Set.<string>}
  249. */
  250. const localeOrder = new Set();
  251. for (const locale of preferredLocales) {
  252. localeOrder.add(locale);
  253. localeOrder.add(LanguageUtils.getBase(locale));
  254. const siblings = shaka.util.Iterables.filter(
  255. localizations.keys(),
  256. (other) => LanguageUtils.areSiblings(other, locale));
  257. // Sort the siblings so that they will always appear in the same order
  258. // regardless of the order of |localizations|.
  259. siblings.sort();
  260. for (const locale of siblings) {
  261. localeOrder.add(locale);
  262. }
  263. const children = shaka.util.Iterables.filter(
  264. localizations.keys(),
  265. (other) => LanguageUtils.getBase(other) == locale);
  266. // Sort the children so that they will always appear in the same order
  267. // regardless of the order of |localizations|.
  268. children.sort();
  269. for (const locale of children) {
  270. localeOrder.add(locale);
  271. }
  272. }
  273. // Finally we add our fallback (something that should have all expected
  274. // entries).
  275. localeOrder.add(fallbackLocale);
  276. // Add all the sibling maps.
  277. /** @type {!Array.<!Map.<string, string>>} */
  278. const mergeOrder = [];
  279. for (const locale of localeOrder) {
  280. const map = localizations.get(locale);
  281. if (map) {
  282. mergeOrder.push(map);
  283. }
  284. }
  285. // We need to reverse the merge order. We build the order based on most
  286. // preferred to least preferred. However, the merge will work in the
  287. // opposite order so we must reverse our maps so that the most preferred
  288. // options will be applied last.
  289. mergeOrder.reverse();
  290. // Merge all the options into our current map.
  291. this.currentMap_.clear();
  292. for (const map of mergeOrder) {
  293. map.forEach((value, key) => {
  294. this.currentMap_.set(key, value);
  295. });
  296. }
  297. // Go through every key we have and see if any preferred locales are
  298. // missing entries. This will allow app developers to find holes in their
  299. // localizations.
  300. /** @type {!Iterable.<string>} */
  301. const allKeys = this.currentMap_.keys();
  302. /** @type {!Set.<string>} */
  303. const missing = new Set();
  304. for (const locale of this.currentLocales_) {
  305. // Make sure we have a non-null map. The diff will be easier that way.
  306. const map = this.localizations_.get(locale) || new Map();
  307. shaka.ui.Localization.findMissingKeys_(map, allKeys, missing);
  308. }
  309. if (missing.size > 0) {
  310. const data = new Map()
  311. // Make a copy of the preferred locales to avoid leaking references.
  312. .set('locales', Array.from(preferredLocales))
  313. // Because more people like arrays more than sets, convert the set to
  314. // an array.
  315. .set('missing', Array.from(missing));
  316. this.dispatchEvent(new shaka.util.FakeEvent(
  317. shaka.ui.Localization.MISSING_LOCALIZATIONS,
  318. data));
  319. }
  320. }
  321. /**
  322. * Go through a map and add all the keys that are in |keys| but not in
  323. * |map| to |missing|.
  324. *
  325. * @param {!Map.<string, string>} map
  326. * @param {!Iterable.<string>} keys
  327. * @param {!Set.<string>} missing
  328. * @private
  329. */
  330. static findMissingKeys_(map, keys, missing) {
  331. for (const key of keys) {
  332. // Check if the value is missing so that we are sure that it does not
  333. // have a value. We get the value and not just |has| so that a null or
  334. // empty string will fail this check.
  335. if (!map.get(key)) {
  336. missing.add(key);
  337. }
  338. }
  339. }
  340. };
  341. /**
  342. * An enum for how the localization system should resolve conflicts between old
  343. * translations and new translations.
  344. *
  345. * @enum {number}
  346. * @export
  347. */
  348. shaka.ui.Localization.ConflictResolution = {
  349. 'USE_OLD': 0,
  350. 'USE_NEW': 1,
  351. };
  352. /**
  353. * The event name for when locales were requested, but we could not find any
  354. * entries for them. The localization system will continue to use the closest
  355. * matches it has.
  356. *
  357. * @const {string}
  358. * @export
  359. */
  360. shaka.ui.Localization.UNKNOWN_LOCALES = 'unknown-locales';
  361. /**
  362. * The event name for when an entry could not be found in the preferred locale,
  363. * related locales, or the fallback locale.
  364. *
  365. * @const {string}
  366. * @export
  367. */
  368. shaka.ui.Localization.UNKNOWN_LOCALIZATION = 'unknown-localization';
  369. /**
  370. * The event name for when entries are missing from the user's preferred
  371. * locale, but we were able to find an entry in a related locale or the fallback
  372. * locale.
  373. *
  374. * @const {string}
  375. * @export
  376. */
  377. shaka.ui.Localization.MISSING_LOCALIZATIONS = 'missing-localizations';
  378. /**
  379. * The event name for when a new locale has been requested and any previously
  380. * resolved values should be updated.
  381. *
  382. * @const {string}
  383. * @export
  384. */
  385. shaka.ui.Localization.LOCALE_CHANGED = 'locale-changed';
  386. /**
  387. * The event name for when |insert| was called and it changed entries that could
  388. * affect previously resolved values.
  389. *
  390. * @const {string}
  391. * @export
  392. */
  393. shaka.ui.Localization.LOCALE_UPDATED = 'locale-updated';
  394. /**
  395. * @event shaka.ui.Localization.UnknownLocalesEvent
  396. * @property {string} type
  397. * 'unknown-locales'
  398. * @property {!Array.<string>} locales
  399. * The locales that the user wanted but could not be found.
  400. * @exportDoc
  401. */
  402. /**
  403. * @event shaka.ui.Localization.MissingLocalizationsEvent
  404. * @property {string} type
  405. * 'unknown-localization'
  406. * @property {!Array.<string>} locales
  407. * The locales that the user wanted.
  408. * @property {string} missing
  409. * The id of the unknown entry.
  410. * @exportDoc
  411. */
  412. /**
  413. * @event shaka.ui.Localization.MissingLocalizationsEvent
  414. * @property {string} type
  415. * 'missing-localizations'
  416. * @property {string} locale
  417. * The locale that the user wanted.
  418. * @property {!Array.<string>} missing
  419. * The ids of the missing entries.
  420. * @exportDoc
  421. */
  422. /**
  423. * @event shaka.ui.Localization.LocaleChangedEvent
  424. * @property {string} type
  425. * 'locale-changed'
  426. * @property {!Array.<string>} locales
  427. * The new set of locales that user wanted,
  428. * and that were successfully found.
  429. * @exportDoc
  430. */