1 | ;
|
2 |
|
3 | Object.defineProperty(exports, "__esModule", {
|
4 | value: true
|
5 | });
|
6 | exports.default = exports.HydrationStateToken = void 0;
|
7 |
|
8 | var _fusionTokens = require("fusion-tokens");
|
9 |
|
10 | var _fusionCore = require("fusion-core");
|
11 |
|
12 | var _fusionPluginUniversalEvents = require("fusion-plugin-universal-events");
|
13 |
|
14 | /** Copyright (c) 2018 Uber Technologies, Inc.
|
15 | *
|
16 | * This source code is licensed under the MIT license found in the
|
17 | * LICENSE file in the root directory of this source tree.
|
18 | *
|
19 | *
|
20 | */
|
21 |
|
22 | /* eslint-env browser */
|
23 | function loadTranslations() {
|
24 | const element = document.getElementById('__TRANSLATIONS__');
|
25 |
|
26 | if (!element) {
|
27 | throw new Error('[fusion-plugin-i18n] - Could not find a __TRANSLATIONS__ element');
|
28 | }
|
29 |
|
30 | try {
|
31 | return JSON.parse((0, _fusionCore.unescape)(element.textContent));
|
32 | } catch (e) {
|
33 | throw new Error('[fusion-plugin-i18n] - Error parsing __TRANSLATIONS__ element content');
|
34 | }
|
35 | }
|
36 |
|
37 | const HydrationStateToken = (0, _fusionCore.createToken)('HydrationStateToken');
|
38 | exports.HydrationStateToken = HydrationStateToken;
|
39 |
|
40 | const pluginFactory = () => (0, _fusionCore.createPlugin)({
|
41 | deps: {
|
42 | fetch: _fusionTokens.FetchToken.optional,
|
43 | hydrationState: HydrationStateToken.optional,
|
44 | events: _fusionPluginUniversalEvents.UniversalEventsToken.optional
|
45 | },
|
46 | provides: ({
|
47 | fetch = window.fetch,
|
48 | hydrationState,
|
49 | events
|
50 | } = {}) => {
|
51 | class I18n {
|
52 | constructor() {
|
53 | const {
|
54 | localeCode,
|
55 | translations
|
56 | } = hydrationState || loadTranslations();
|
57 | this.requestedKeys = new Set();
|
58 | this.translations = translations || {};
|
59 |
|
60 | if (localeCode) {
|
61 | this.locale = localeCode;
|
62 | }
|
63 | }
|
64 |
|
65 | async load(translationKeys) {
|
66 | const loadedKeys = Object.keys(this.translations);
|
67 | const unloaded = translationKeys.filter(key => {
|
68 | return loadedKeys.indexOf(key) < 0 && !this.requestedKeys.has(key);
|
69 | });
|
70 |
|
71 | if (unloaded.length > 0) {
|
72 | // Don't try to load translations again if a request is already in
|
73 | // flight. This means that we need to add unloaded chunks to
|
74 | // loadedChunks optimistically and remove them if some error happens
|
75 | unloaded.forEach(key => {
|
76 | this.requestedKeys.add(key);
|
77 | });
|
78 | const fetchOpts = {
|
79 | method: 'POST',
|
80 | headers: {
|
81 | Accept: '*/*',
|
82 | 'Content-Type': 'application/json',
|
83 | ...(this.locale ? {
|
84 | 'X-Fusion-Locale-Code': this.locale
|
85 | } : {})
|
86 | },
|
87 | body: JSON.stringify(unloaded)
|
88 | }; // TODO(#3) don't append prefix if injected fetch also injects prefix
|
89 |
|
90 | return fetch(`/_translations${this.locale ? `?localeCode=${this.locale}` : ''}`, fetchOpts).then(r => {
|
91 | try {
|
92 | return r.json();
|
93 | } catch (err) {
|
94 | events && events.emit('i18n-load-error', {
|
95 | text: r.text()
|
96 | });
|
97 | throw err;
|
98 | }
|
99 | }).then(data => {
|
100 | for (const key in data) {
|
101 | this.translations[key] = data[key];
|
102 | this.requestedKeys.delete(key);
|
103 | }
|
104 | }).catch(err => {
|
105 | // An error occurred, so remove the chunks we were trying to load
|
106 | // from loadedChunks. This allows us to try to load those chunk
|
107 | // translations again
|
108 | unloaded.forEach(key => {
|
109 | this.requestedKeys.delete(key);
|
110 | });
|
111 | });
|
112 | }
|
113 | }
|
114 |
|
115 | translate(key, interpolations = {}) {
|
116 | const template = this.translations[key];
|
117 |
|
118 | if (typeof template !== 'string') {
|
119 | events && events.emit('i18n-translate-miss', {
|
120 | key
|
121 | });
|
122 | return key;
|
123 | }
|
124 |
|
125 | return template.replace(/\${(.*?)}/g, (_, k) => interpolations[k] === void 0 ? '${' + k + '}' : String(interpolations[k]));
|
126 | }
|
127 |
|
128 | }
|
129 |
|
130 | const i18n = new I18n();
|
131 | return {
|
132 | from: () => i18n
|
133 | };
|
134 | }
|
135 | });
|
136 |
|
137 | var _default = false && pluginFactory();
|
138 |
|
139 | exports.default = _default;
|
140 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["browser.js"],"names":["loadTranslations","element","document","getElementById","Error","JSON","parse","textContent","e","HydrationStateToken","pluginFactory","deps","fetch","FetchToken","optional","hydrationState","events","UniversalEventsToken","provides","window","I18n","constructor","localeCode","translations","requestedKeys","Set","locale","load","translationKeys","loadedKeys","Object","keys","unloaded","filter","key","indexOf","has","length","forEach","add","fetchOpts","method","headers","Accept","body","stringify","then","r","json","err","emit","text","data","delete","catch","translate","interpolations","template","replace","_","k","String","i18n","from"],"mappings":"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","sourcesContent":["/** Copyright (c) 2018 Uber Technologies, Inc.\n *\n * This source code is licensed under the MIT license found in the\n * LICENSE file in the root directory of this source tree.\n *\n * @flow\n */\n\n/* eslint-env browser */\nimport {FetchToken} from 'fusion-tokens';\nimport {createPlugin, unescape, createToken} from 'fusion-core';\nimport type {FusionPlugin, Token} from 'fusion-core';\nimport {UniversalEventsToken} from 'fusion-plugin-universal-events';\n\nimport type {\n  I18nDepsType,\n  I18nServiceType,\n  TranslationsObjectType,\n} from './types.js';\n\ntype LoadedTranslationsType = {\n  localeCode?: string,\n  translations?: TranslationsObjectType,\n};\nfunction loadTranslations(): LoadedTranslationsType {\n  const element = document.getElementById('__TRANSLATIONS__');\n  if (!element) {\n    throw new Error(\n      '[fusion-plugin-i18n] - Could not find a __TRANSLATIONS__ element'\n    );\n  }\n  try {\n    return JSON.parse(unescape(element.textContent));\n  } catch (e) {\n    throw new Error(\n      '[fusion-plugin-i18n] - Error parsing __TRANSLATIONS__ element content'\n    );\n  }\n}\n\ntype HydrationStateType = {\n  localeCode?: string,\n  translations: TranslationsObjectType,\n};\nexport const HydrationStateToken: Token<HydrationStateType> = createToken(\n  'HydrationStateToken'\n);\n\ntype PluginType = FusionPlugin<I18nDepsType, I18nServiceType>;\nconst pluginFactory: () => PluginType = () =>\n  createPlugin({\n    deps: {\n      fetch: FetchToken.optional,\n      hydrationState: HydrationStateToken.optional,\n      events: UniversalEventsToken.optional,\n    },\n    provides: ({fetch = window.fetch, hydrationState, events} = {}) => {\n      class I18n {\n        locale: string;\n        translations: TranslationsObjectType;\n        requestedKeys: Set<string>;\n\n        constructor() {\n          const {localeCode, translations} =\n            hydrationState || loadTranslations();\n          this.requestedKeys = new Set();\n          this.translations = translations || {};\n          if (localeCode) {\n            this.locale = localeCode;\n          }\n        }\n        async load(translationKeys) {\n          const loadedKeys = Object.keys(this.translations);\n          const unloaded = translationKeys.filter(key => {\n            return loadedKeys.indexOf(key) < 0 && !this.requestedKeys.has(key);\n          });\n          if (unloaded.length > 0) {\n            // Don't try to load translations again if a request is already in\n            // flight. This means that we need to add unloaded chunks to\n            // loadedChunks optimistically and remove them if some error happens\n            unloaded.forEach(key => {\n              this.requestedKeys.add(key);\n            });\n            const fetchOpts = {\n              method: 'POST',\n              headers: {\n                Accept: '*/*',\n                'Content-Type': 'application/json',\n                ...(this.locale ? {'X-Fusion-Locale-Code': this.locale} : {}),\n              },\n              body: JSON.stringify(unloaded),\n            };\n            // TODO(#3) don't append prefix if injected fetch also injects prefix\n            return fetch(\n              `/_translations${\n                this.locale ? `?localeCode=${this.locale}` : ''\n              }`,\n              fetchOpts\n            )\n              .then(r => {\n                try {\n                  return r.json();\n                } catch (err) {\n                  events && events.emit('i18n-load-error', {text: r.text()});\n                  throw err;\n                }\n              })\n              .then((data: {[string]: string}) => {\n                for (const key in data) {\n                  this.translations[key] = data[key];\n                  this.requestedKeys.delete(key);\n                }\n              })\n              .catch((err: Error) => {\n                // An error occurred, so remove the chunks we were trying to load\n                // from loadedChunks. This allows us to try to load those chunk\n                // translations again\n                unloaded.forEach(key => {\n                  this.requestedKeys.delete(key);\n                });\n              });\n          }\n        }\n        translate(key, interpolations = {}) {\n          const template = this.translations[key];\n\n          if (typeof template !== 'string') {\n            events && events.emit('i18n-translate-miss', {key});\n            return key;\n          }\n\n          return template.replace(/\\${(.*?)}/g, (_, k) =>\n            interpolations[k] === void 0\n              ? '${' + k + '}'\n              : String(interpolations[k])\n          );\n        }\n      }\n      const i18n = new I18n();\n      return {from: () => i18n};\n    },\n  });\n\nexport default ((__BROWSER__ && pluginFactory(): any): PluginType);\n"]} |
\ | No newline at end of file |