UNPKG

@spartacus/core

Version:

Spartacus - the core framework

100 lines 15.7 kB
import { Injectable } from '@angular/core'; import i18nextHttpBackend from 'i18next-http-backend'; import { tap } from 'rxjs/operators'; import * as i0 from "@angular/core"; export function i18nextInit(i18next, configInit, languageService, httpClient, serverRequestOrigin, siteContextI18nextSynchronizer) { return () => configInit .getStable('i18n') .pipe(tap((config) => { var _a, _b, _c, _d; let i18nextConfig = { ns: [], fallbackLng: (_a = config.i18n) === null || _a === void 0 ? void 0 : _a.fallbackLang, debug: (_b = config.i18n) === null || _b === void 0 ? void 0 : _b.debug, interpolation: { escapeValue: false, }, }; if ((_d = (_c = config.i18n) === null || _c === void 0 ? void 0 : _c.backend) === null || _d === void 0 ? void 0 : _d.loadPath) { i18next = i18next.use(i18nextHttpBackend); const loadPath = getLoadPath(config.i18n.backend.loadPath, serverRequestOrigin); const backend = { loadPath, request: i18nextGetHttpClient(httpClient), // Disable the periodical reloading. Otherwise SSR would not finish due to the pending task `setInterval()` // See source code of `i18next-http-backend` : https://github.com/i18next/i18next-http-backend/blob/00b7e8f67abf8372af17529b51190a7e8b17e3d8/lib/index.js#L40-L41 reloadInterval: false, }; i18nextConfig = Object.assign(Object.assign({}, i18nextConfig), { backend }); } return i18next.init(i18nextConfig, () => { var _a; // Don't use i18next's 'resources' config key for adding static translations, // because it will disable loading chunks from backend. We add resources here, in the init's callback. i18nextAddTranslations(i18next, (_a = config.i18n) === null || _a === void 0 ? void 0 : _a.resources); siteContextI18nextSynchronizer.init(i18next, languageService); }); })) .toPromise(); } export function i18nextAddTranslations(i18next, resources = {}) { Object.keys(resources).forEach((lang) => { Object.keys(resources[lang]).forEach((chunkName) => { i18next.addResourceBundle(lang, chunkName, resources[lang][chunkName], true, true); }); }); } export class SiteContextI18nextSynchronizer { init(i18next, language) { var _a; // always update language of i18next on site context (language) change this.sub = (_a = this.sub) !== null && _a !== void 0 ? _a : language.getActive().subscribe((lang) => i18next.changeLanguage(lang)); } ngOnDestroy() { var _a; (_a = this.sub) === null || _a === void 0 ? void 0 : _a.unsubscribe(); } } SiteContextI18nextSynchronizer.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: SiteContextI18nextSynchronizer, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); SiteContextI18nextSynchronizer.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: SiteContextI18nextSynchronizer, providedIn: 'root' }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.5", ngImport: i0, type: SiteContextI18nextSynchronizer, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }] }); /** * Returns a function appropriate for i18next to make http calls for JSON files. * See docs for `i18next-http-backend`: https://github.com/i18next/i18next-http-backend#backend-options * * It uses Angular HttpClient under the hood, so it works in SSR. * @param httpClient Angular http client */ export function i18nextGetHttpClient(httpClient) { return (_options, url, _payload, callback) => { httpClient.get(url, { responseType: 'text' }).subscribe((data) => callback(null, { status: 200, data }), (error) => callback(error, { // a workaround for https://github.com/i18next/i18next-http-backend/issues/82 data: null, status: error.status, })); }; } /** * Resolves the relative path to the absolute one in SSR, using the server request's origin. * It's needed, because Angular Universal doesn't support relative URLs in HttpClient. See Angular issues: * - https://github.com/angular/angular/issues/19224 * - https://github.com/angular/universal/issues/858 */ export function getLoadPath(path, serverRequestOrigin) { if (serverRequestOrigin && !path.match(/^http(s)?:\/\//)) { if (path.startsWith('/')) { path = path.slice(1); } if (path.startsWith('./')) { path = path.slice(2); } const result = `${serverRequestOrigin}/${path}`; return result; } return path; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18next-init.js","sourceRoot":"","sources":["../../../../../../projects/core/src/i18n/i18next/i18next-init.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient } from '@angular/common/http';\nimport { Injectable, OnDestroy } from '@angular/core';\nimport { i18n, InitOptions } from 'i18next';\nimport i18nextHttpBackend, {\n  BackendOptions,\n  RequestCallback,\n} from 'i18next-http-backend';\nimport { Subscription } from 'rxjs';\nimport { tap } from 'rxjs/operators';\nimport { ConfigInitializerService } from '../../config/config-initializer/config-initializer.service';\nimport { LanguageService } from '../../site-context/facade/language.service';\nimport { TranslationResources } from '../translation-resources';\n\nexport function i18nextInit(\n  i18next: i18n,\n  configInit: ConfigInitializerService,\n  languageService: LanguageService,\n  httpClient: HttpClient,\n  serverRequestOrigin: string | null,\n  siteContextI18nextSynchronizer: SiteContextI18nextSynchronizer\n): () => Promise<any> {\n  return () =>\n    configInit\n      .getStable('i18n')\n      .pipe(\n        tap((config) => {\n          let i18nextConfig: InitOptions = {\n            ns: [], // don't preload any namespaces\n            fallbackLng: config.i18n?.fallbackLang,\n            debug: config.i18n?.debug,\n            interpolation: {\n              escapeValue: false,\n            },\n          };\n\n          if (config.i18n?.backend?.loadPath) {\n            i18next = i18next.use(i18nextHttpBackend);\n            const loadPath = getLoadPath(\n              config.i18n.backend.loadPath,\n              serverRequestOrigin\n            );\n            const backend: BackendOptions = {\n              loadPath,\n              request: i18nextGetHttpClient(httpClient),\n\n              // Disable the periodical reloading. Otherwise SSR would not finish due to the pending task `setInterval()`\n              // See source code of `i18next-http-backend` : https://github.com/i18next/i18next-http-backend/blob/00b7e8f67abf8372af17529b51190a7e8b17e3d8/lib/index.js#L40-L41\n              reloadInterval: false,\n            };\n            i18nextConfig = { ...i18nextConfig, backend };\n          }\n\n          return i18next.init(i18nextConfig, () => {\n            // Don't use i18next's 'resources' config key for adding static translations,\n            // because it will disable loading chunks from backend. We add resources here, in the init's callback.\n            i18nextAddTranslations(i18next, config.i18n?.resources);\n            siteContextI18nextSynchronizer.init(i18next, languageService);\n          });\n        })\n      )\n      .toPromise();\n}\n\nexport function i18nextAddTranslations(\n  i18next: i18n,\n  resources: TranslationResources = {}\n): void {\n  Object.keys(resources).forEach((lang) => {\n    Object.keys(resources[lang]).forEach((chunkName) => {\n      i18next.addResourceBundle(\n        lang,\n        chunkName,\n        resources[lang][chunkName],\n        true,\n        true\n      );\n    });\n  });\n}\n\n@Injectable({ providedIn: 'root' })\nexport class SiteContextI18nextSynchronizer implements OnDestroy {\n  sub: Subscription;\n\n  init(i18next: i18n, language: LanguageService) {\n    // always update language of i18next on site context (language) change\n    this.sub =\n      this.sub ??\n      language.getActive().subscribe((lang) => i18next.changeLanguage(lang));\n  }\n\n  ngOnDestroy() {\n    this.sub?.unsubscribe();\n  }\n}\n\n/**\n * Returns a function appropriate for i18next to make http calls for JSON files.\n * See docs for `i18next-http-backend`: https://github.com/i18next/i18next-http-backend#backend-options\n *\n * It uses Angular HttpClient under the hood, so it works in SSR.\n * @param httpClient Angular http client\n */\nexport function i18nextGetHttpClient(\n  httpClient: HttpClient\n): (\n  options: BackendOptions,\n  url: string,\n  payload: object | string,\n  callback: RequestCallback\n) => void {\n  return (\n    _options: BackendOptions,\n    url: string,\n    _payload: object | string,\n    callback: RequestCallback\n  ) => {\n    httpClient.get(url, { responseType: 'text' }).subscribe(\n      (data) => callback(null, { status: 200, data }),\n      (error) =>\n        callback(error, {\n          // a workaround for https://github.com/i18next/i18next-http-backend/issues/82\n          data: null as any,\n          status: error.status,\n        })\n    );\n  };\n}\n\n/**\n * Resolves the relative path to the absolute one in SSR, using the server request's origin.\n * It's needed, because Angular Universal doesn't support relative URLs in HttpClient. See Angular issues:\n * - https://github.com/angular/angular/issues/19224\n * - https://github.com/angular/universal/issues/858\n */\nexport function getLoadPath(\n  path: string,\n  serverRequestOrigin: string | null\n): string {\n  if (serverRequestOrigin && !path.match(/^http(s)?:\\/\\//)) {\n    if (path.startsWith('/')) {\n      path = path.slice(1);\n    }\n    if (path.startsWith('./')) {\n      path = path.slice(2);\n    }\n    const result = `${serverRequestOrigin}/${path}`;\n    return result;\n  }\n  return path;\n}\n"]}