Source: ui/localization.js

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