@tolgee/core 4.7.0 → 4.7.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (97) hide show
  1. package/dist/tolgee.cjs.js +2 -4
  2. package/dist/tolgee.cjs.js.map +1 -1
  3. package/dist/tolgee.cjs.min.js +1 -1
  4. package/dist/tolgee.cjs.min.js.map +1 -1
  5. package/dist/{tolgee.esm.js → tolgee.esm.min.mjs} +2 -2
  6. package/dist/tolgee.esm.min.mjs.map +1 -0
  7. package/dist/tolgee.esm.mjs +5690 -0
  8. package/dist/tolgee.esm.mjs.map +1 -0
  9. package/dist/tolgee.umd.js +2 -4
  10. package/dist/tolgee.umd.js.map +1 -1
  11. package/dist/tolgee.umd.min.js +1 -1
  12. package/dist/tolgee.umd.min.js.map +1 -1
  13. package/package.json +10 -9
  14. package/src/Constants/Global.ts +9 -0
  15. package/src/Constants/ModifierKey.ts +6 -0
  16. package/src/Errors/ApiHttpError.ts +8 -0
  17. package/src/Observer.test.ts +119 -0
  18. package/src/Observer.ts +68 -0
  19. package/src/Properties.test.ts +150 -0
  20. package/src/Properties.ts +112 -0
  21. package/src/Tolgee.test.ts +473 -0
  22. package/src/Tolgee.ts +335 -0
  23. package/src/TolgeeConfig.test.ts +21 -0
  24. package/src/TolgeeConfig.ts +134 -0
  25. package/src/__integration/FormatterIcu.test.ts +80 -0
  26. package/src/__integration/FormatterMissing.ts +54 -0
  27. package/src/__integration/Tolgee.test.ts +90 -0
  28. package/src/__integration/TolgeeInvisible.test.ts +145 -0
  29. package/src/__integration/mockTranslations.ts +6 -0
  30. package/src/__integration/testConfig.ts +16 -0
  31. package/src/__testFixtures/classMock.ts +11 -0
  32. package/src/__testFixtures/createElement.ts +43 -0
  33. package/src/__testFixtures/createTestDom.ts +25 -0
  34. package/src/__testFixtures/mocked.ts +25 -0
  35. package/src/__testFixtures/setupAfterEnv.ts +34 -0
  36. package/src/helpers/NodeHelper.ts +90 -0
  37. package/src/helpers/TextHelper.test.ts +62 -0
  38. package/src/helpers/TextHelper.ts +58 -0
  39. package/src/helpers/commonTypes.ts +8 -0
  40. package/src/helpers/encoderPolyfill.ts +96 -0
  41. package/src/helpers/secret.test.ts +61 -0
  42. package/src/helpers/secret.ts +68 -0
  43. package/src/helpers/sleep.ts +2 -0
  44. package/src/highlighter/HighlightFunctionsInitializer.test.ts +40 -0
  45. package/src/highlighter/HighlightFunctionsInitializer.ts +61 -0
  46. package/src/highlighter/MouseEventHandler.test.ts +151 -0
  47. package/src/highlighter/MouseEventHandler.ts +191 -0
  48. package/src/highlighter/TranslationHighlighter.test.ts +177 -0
  49. package/src/highlighter/TranslationHighlighter.ts +113 -0
  50. package/src/index.ts +10 -0
  51. package/src/internal.ts +2 -0
  52. package/src/modules/IcuFormatter.ts +17 -0
  53. package/src/modules/index.ts +1 -0
  54. package/src/services/ApiHttpService.ts +85 -0
  55. package/src/services/CoreService.test.ts +142 -0
  56. package/src/services/CoreService.ts +76 -0
  57. package/src/services/DependencyService.test.ts +51 -0
  58. package/src/services/DependencyService.ts +116 -0
  59. package/src/services/ElementRegistrar.test.ts +131 -0
  60. package/src/services/ElementRegistrar.ts +108 -0
  61. package/src/services/EventEmitter.ts +52 -0
  62. package/src/services/EventService.ts +14 -0
  63. package/src/services/ModuleService.ts +14 -0
  64. package/src/services/ScreenshotService.ts +31 -0
  65. package/src/services/Subscription.ts +7 -0
  66. package/src/services/TextService.test.ts +88 -0
  67. package/src/services/TextService.ts +82 -0
  68. package/src/services/TranslationService.test.ts +358 -0
  69. package/src/services/TranslationService.ts +417 -0
  70. package/src/services/__mocks__/CoreService.ts +17 -0
  71. package/src/toolsManager/Messages.test.ts +79 -0
  72. package/src/toolsManager/Messages.ts +60 -0
  73. package/src/toolsManager/PluginManager.test.ts +108 -0
  74. package/src/toolsManager/PluginManager.ts +129 -0
  75. package/src/types/DTOs.ts +25 -0
  76. package/src/types/apiSchema.generated.ts +6208 -0
  77. package/src/types.ts +146 -0
  78. package/src/wrappers/AbstractWrapper.ts +14 -0
  79. package/src/wrappers/NodeHandler.ts +143 -0
  80. package/src/wrappers/WrappedHandler.ts +28 -0
  81. package/src/wrappers/invisible/AttributeHandler.ts +23 -0
  82. package/src/wrappers/invisible/Coder.ts +65 -0
  83. package/src/wrappers/invisible/ContentHandler.ts +15 -0
  84. package/src/wrappers/invisible/CoreHandler.ts +17 -0
  85. package/src/wrappers/invisible/InvisibleWrapper.ts +59 -0
  86. package/src/wrappers/invisible/ValueMemory.test.ts +25 -0
  87. package/src/wrappers/invisible/ValueMemory.ts +16 -0
  88. package/src/wrappers/text/AttributeHandler.test.ts +117 -0
  89. package/src/wrappers/text/AttributeHandler.ts +25 -0
  90. package/src/wrappers/text/Coder.test.ts +298 -0
  91. package/src/wrappers/text/Coder.ts +202 -0
  92. package/src/wrappers/text/ContentHandler.test.ts +185 -0
  93. package/src/wrappers/text/ContentHandler.ts +21 -0
  94. package/src/wrappers/text/CoreHandler.test.ts +106 -0
  95. package/src/wrappers/text/CoreHandler.ts +45 -0
  96. package/src/wrappers/text/TextWrapper.ts +69 -0
  97. package/dist/tolgee.esm.js.map +0 -1
package/package.json CHANGED
@@ -1,9 +1,9 @@
1
1
  {
2
2
  "name": "@tolgee/core",
3
- "version": "4.7.0",
3
+ "version": "4.7.2",
4
4
  "description": "Library providing ability to translate messages directly in context of developed application.",
5
5
  "main": "index.js",
6
- "module": "./dist/tolgee.esm.js",
6
+ "module": "./dist/tolgee.esm.mjs",
7
7
  "types": "./lib/index.d.ts",
8
8
  "repository": {
9
9
  "type": "git",
@@ -37,7 +37,8 @@
37
37
  "files": [
38
38
  "index.js",
39
39
  "lib/**/*",
40
- "dist/**/*"
40
+ "dist/**/*",
41
+ "src/**/*"
41
42
  ],
42
43
  "devDependencies": {
43
44
  "@rollup/plugin-node-resolve": "13.3.0",
@@ -48,7 +49,7 @@
48
49
  "@types/intl-messageformat": "^3.0.0",
49
50
  "@types/jest": "^27.0.2",
50
51
  "@types/node": "^17.0.8",
51
- "concurrently": "7.1.0",
52
+ "concurrently": "7.2.2",
52
53
  "intl-messageformat": "^9.9.1",
53
54
  "jest": "^27.2.4",
54
55
  "jest-fetch-mock": "^3.0.3",
@@ -56,16 +57,16 @@
56
57
  "rollup": "^2.56.3",
57
58
  "rollup-plugin-terser": "7.0.2",
58
59
  "ts-jest": "^27.0.5",
59
- "tslib": "^2.3.1",
60
+ "tslib": "^2.4.0",
60
61
  "typedoc": "0.22.15",
61
- "typedoc-plugin-markdown": "3.12.1",
62
- "typescript": "4.6.4"
62
+ "typedoc-plugin-markdown": "3.13.2",
63
+ "typescript": "4.7.4"
63
64
  },
64
65
  "exports": {
65
66
  "require": "./dist/tolgee.cjs.js",
66
- "import": "./dist/tolgee.esm.js"
67
+ "import": "./dist/tolgee.esm.mjs"
67
68
  },
68
- "gitHead": "068b8450eb29e1efd173eaa5537e44504a390ee3",
69
+ "gitHead": "12ccff7336da87314feac6cdff5ea19c08fe15d7",
69
70
  "publishConfig": {
70
71
  "access": "public"
71
72
  }
@@ -0,0 +1,9 @@
1
+ export const RESTRICTED_ASCENDANT_ATTRIBUTE = 'data-tolgee-restricted';
2
+
3
+ export const TOLGEE_ATTRIBUTE_NAME = '_tolgee';
4
+ export const TOLGEE_TARGET_ATTRIBUTE = '_tolgee-target';
5
+ export const TOLGEE_WRAPPED_ONLY_DATA_ATTRIBUTE = 'data-tolgee-key-only';
6
+ export const TOLGEE_HIGHLIGHTER_CLASS = '_tolgee-highlighter';
7
+
8
+ // needs to be same as in @tolgee/ui package
9
+ export const DEVTOOLS_ID = '__tolgee_dev_tools';
@@ -0,0 +1,6 @@
1
+ export enum ModifierKey {
2
+ Alt,
3
+ Control,
4
+ Shift,
5
+ Meta,
6
+ }
@@ -0,0 +1,8 @@
1
+ export class ApiHttpError extends Error {
2
+ constructor(public response: Response, public code?: string) {
3
+ super('Api http error');
4
+
5
+ // Set the prototype explicitly.
6
+ Object.setPrototypeOf(this, ApiHttpError.prototype);
7
+ }
8
+ }
@@ -0,0 +1,119 @@
1
+ jest.dontMock('./Observer');
2
+ jest.dontMock('./services/DependencyService');
3
+
4
+ import { DependencyService } from './services/DependencyService';
5
+ import '@testing-library/jest-dom/extend-expect';
6
+ import 'regenerator-runtime/runtime.js';
7
+ import { getMockedInstance } from '@testFixtures/mocked';
8
+ import { Properties } from './Properties';
9
+ import { waitFor } from '@testing-library/dom';
10
+ import { Observer } from './Observer';
11
+ import { ElementRegistrar } from './services/ElementRegistrar';
12
+ import { TextWrapper } from './wrappers/text/TextWrapper';
13
+
14
+ describe('Observer', () => {
15
+ let observer: Observer;
16
+ let properties: Properties;
17
+
18
+ beforeEach(async () => {
19
+ const dependencyService = new DependencyService();
20
+ dependencyService.init({ wrapperMode: 'text' });
21
+ dependencyService.run();
22
+ observer = dependencyService.observer;
23
+ properties = getMockedInstance(Properties);
24
+ document.body = document.createElement('body');
25
+ properties.config.targetElement = document.body;
26
+ });
27
+
28
+ afterEach(() => {
29
+ jest.clearAllMocks();
30
+ });
31
+
32
+ test('Can be created', () => {
33
+ expect(observer).toBeInstanceOf(Observer);
34
+ });
35
+
36
+ describe('Tests on document.body', () => {
37
+ test('Will handle mutation on basic character data', async () => {
38
+ const text = document.createTextNode('Dummy text node');
39
+ document.body.append(text);
40
+ observer.observe();
41
+ text.textContent = 'Dummy text node modified';
42
+
43
+ await waitFor(() => {
44
+ const onNewNodesMock = getMockedInstance(TextWrapper).handleText;
45
+ expect(onNewNodesMock).toBeCalledTimes(1);
46
+ expect(onNewNodesMock).toBeCalledWith(text);
47
+ });
48
+ });
49
+
50
+ test('Will handle mutation element child list', async () => {
51
+ const text = document.createTextNode('Dummy text node');
52
+ observer.observe();
53
+ document.body.append(text);
54
+
55
+ await waitFor(() => {
56
+ const handleSubtree = getMockedInstance(TextWrapper).handleSubtree;
57
+
58
+ expect(handleSubtree).toBeCalledTimes(1);
59
+ expect(handleSubtree).toBeCalledWith(document.body);
60
+ });
61
+ });
62
+
63
+ test('Will handle attribute mutation', async () => {
64
+ const span = document.createElement('span');
65
+ const attributeName = 'data-attribute';
66
+ span.setAttribute(attributeName, 'text');
67
+ document.body.append(span);
68
+ observer.observe();
69
+ span.setAttribute(attributeName, 'modified');
70
+
71
+ await waitFor(() => {
72
+ const handleAttributeMock =
73
+ getMockedInstance(TextWrapper).handleAttribute;
74
+ expect(handleAttributeMock).toBeCalledTimes(1);
75
+ expect(handleAttributeMock).toBeCalledWith(span);
76
+ });
77
+ });
78
+
79
+ test('Will handle subtree mutation', async () => {
80
+ const div = document.createElement('div');
81
+ div.innerHTML = "<div><div></div><div id='innerDiv'></div></div>";
82
+ document.body.append(div);
83
+ const innerDiv = document.getElementById('innerDiv');
84
+ observer.observe();
85
+ innerDiv.textContent = 'This is inner text';
86
+
87
+ await waitFor(() => {
88
+ const handleAttributeMock =
89
+ getMockedInstance(TextWrapper).handleSubtree;
90
+
91
+ expect(handleAttributeMock).toBeCalledTimes(1);
92
+ expect(handleAttributeMock).toBeCalledWith(innerDiv);
93
+ });
94
+ });
95
+
96
+ test('will stop observing', async () => {
97
+ observer.observe();
98
+ observer.stopObserving();
99
+
100
+ document.body.textContent = 'Nothing';
101
+ await new Promise((resolve) => setTimeout(resolve, 10));
102
+
103
+ const handleSubtree = getMockedInstance(TextWrapper).handleSubtree;
104
+ expect(handleSubtree).not.toBeCalledTimes(1);
105
+ });
106
+ });
107
+
108
+ test("Will call registrar's cleanInactive on change", async () => {
109
+ const text = document.createTextNode('Dummy text node');
110
+ document.body.append(text);
111
+ observer.observe();
112
+ text.textContent = 'Dummy text node modified';
113
+
114
+ await waitFor(() => {
115
+ const refreshAllMock = getMockedInstance(ElementRegistrar).refreshAll;
116
+ expect(refreshAllMock).toBeCalledTimes(1);
117
+ });
118
+ });
119
+ });
@@ -0,0 +1,68 @@
1
+ import { Properties } from './Properties';
2
+ import { ElementRegistrar } from './services/ElementRegistrar';
3
+ import { AbstractWrapper } from './wrappers/AbstractWrapper';
4
+
5
+ export class Observer {
6
+ constructor(
7
+ private properties: Properties,
8
+ private textWrapper: AbstractWrapper,
9
+ private nodeRegistrar: ElementRegistrar
10
+ ) {}
11
+
12
+ private _observer = undefined;
13
+ private _observing = false;
14
+
15
+ private get observer(): MutationObserver | undefined {
16
+ if (!this._observer && typeof window !== 'undefined') {
17
+ this._observer = new MutationObserver(
18
+ async (mutationsList: MutationRecord[]) => {
19
+ for (const mutation of mutationsList) {
20
+ if (!this._observing) {
21
+ // make sure we don't touch the DOM after disconnect is called
22
+ return;
23
+ }
24
+ if (mutation.type === 'characterData') {
25
+ await this.textWrapper.handleText(mutation.target as Element);
26
+ continue;
27
+ }
28
+ if (mutation.type === 'childList') {
29
+ await this.textWrapper.handleSubtree(mutation.target as Element);
30
+ continue;
31
+ }
32
+ if (mutation.type === 'attributes') {
33
+ await this.textWrapper.handleAttribute(
34
+ mutation.target as Element
35
+ );
36
+ }
37
+ }
38
+ this.nodeRegistrar.refreshAll();
39
+ }
40
+ );
41
+ }
42
+ return this._observer;
43
+ }
44
+
45
+ public observe() {
46
+ if (!this.observer) {
47
+ return;
48
+ }
49
+ if (this._observing) {
50
+ return;
51
+ }
52
+ this._observing = true;
53
+ this.observer.observe(this.properties.config.targetElement, {
54
+ attributes: true,
55
+ childList: true,
56
+ subtree: true,
57
+ characterData: true,
58
+ });
59
+ }
60
+
61
+ public stopObserving() {
62
+ if (!this.observer) {
63
+ return;
64
+ }
65
+ this._observing = false;
66
+ this.observer.disconnect();
67
+ }
68
+ }
@@ -0,0 +1,150 @@
1
+ jest.dontMock('./Properties');
2
+ jest.dontMock('./TolgeeConfig');
3
+
4
+ import '@testing-library/jest-dom/extend-expect';
5
+ import { mocked } from 'ts-jest/utils';
6
+ import { TolgeeConfig } from './TolgeeConfig';
7
+ import { Properties } from './Properties';
8
+
9
+ describe('Properties', () => {
10
+ let properties: Properties;
11
+
12
+ beforeEach(() => {
13
+ jest.clearAllMocks();
14
+ properties = new Properties();
15
+ });
16
+
17
+ test('can be created', () => {
18
+ expect(properties).not.toBeNull();
19
+ });
20
+
21
+ describe('preferred languages', () => {
22
+ test('getter returns from local storage', () => {
23
+ const dummyReturn = ['dummyLang1'];
24
+ Storage.prototype.getItem = jest.fn();
25
+ mocked(localStorage.getItem).mockReturnValueOnce(
26
+ JSON.stringify(dummyReturn)
27
+ );
28
+ expect(properties.preferredLanguages).toEqual(new Set(dummyReturn));
29
+ expect(localStorage.getItem).toBeCalledWith(
30
+ '__tolgee_preferredLanguages'
31
+ );
32
+ });
33
+
34
+ test('setter sets local storage item', () => {
35
+ const dummySet = ['dummyLang1'];
36
+ Storage.prototype.setItem = jest.fn();
37
+ properties.preferredLanguages = new Set(dummySet);
38
+ expect(localStorage.setItem).toBeCalledWith(
39
+ '__tolgee_preferredLanguages',
40
+ JSON.stringify(dummySet)
41
+ );
42
+ });
43
+ });
44
+
45
+ describe('current language', () => {
46
+ test('getter returns from local storage', () => {
47
+ const dummyReturn = 'cs';
48
+ properties.config = new TolgeeConfig({
49
+ availableLanguages: ['cs', 'en'],
50
+ });
51
+ Storage.prototype.getItem = jest.fn();
52
+ mocked(localStorage.getItem).mockReturnValueOnce(dummyReturn);
53
+ expect(properties.currentLanguage).toEqual(dummyReturn);
54
+ expect(localStorage.getItem).toBeCalledWith('__tolgee_currentLanguage');
55
+ });
56
+
57
+ test('uses defaultLanguage without localStorage and language detection', () => {
58
+ const languageGetter = jest.spyOn(window.navigator, 'language', 'get');
59
+ Storage.prototype.getItem = jest.fn();
60
+ Storage.prototype.setItem = jest.fn();
61
+ const defaultLanguage = 'cs';
62
+ properties.config = new TolgeeConfig({
63
+ defaultLanguage,
64
+ enableLanguageStore: false,
65
+ enableLanguageDetection: false,
66
+ });
67
+ expect(properties.currentLanguage).toEqual(defaultLanguage);
68
+
69
+ properties.currentLanguage = 'de';
70
+ expect(properties.currentLanguage).toEqual('de');
71
+
72
+ expect(localStorage.setItem).not.toBeCalled();
73
+ expect(languageGetter).not.toBeCalledWith('__tolgee_currentLanguage');
74
+ expect(localStorage.getItem).not.toBeCalledWith(
75
+ '__tolgee_currentLanguage'
76
+ );
77
+ });
78
+
79
+ test('setter sets local storage item', () => {
80
+ const dummySet = 'dummyLang1';
81
+ Storage.prototype.setItem = jest.fn();
82
+ properties.config = new TolgeeConfig();
83
+ properties.currentLanguage = dummySet;
84
+ expect(localStorage.setItem).toBeCalledWith(
85
+ '__tolgee_currentLanguage',
86
+ dummySet
87
+ );
88
+ });
89
+
90
+ test('returns correct lang by navigator', () => {
91
+ const languageGetter = jest.spyOn(window.navigator, 'language', 'get');
92
+ languageGetter.mockReturnValue('cs');
93
+ properties.config = new TolgeeConfig({
94
+ availableLanguages: ['en', 'cs'],
95
+ });
96
+ expect(properties.currentLanguage).toEqual('cs');
97
+ });
98
+
99
+ test('language is stored locally and detected only once', () => {
100
+ const languageGetter = jest.spyOn(window.navigator, 'language', 'get');
101
+ languageGetter.mockReturnValue('cs');
102
+ Storage.prototype.getItem = jest.fn();
103
+ properties.config = new TolgeeConfig({
104
+ availableLanguages: ['cs', 'en'],
105
+ });
106
+ expect(properties.currentLanguage).toEqual('cs');
107
+ expect(properties.currentLanguage).toEqual('cs');
108
+
109
+ expect(languageGetter).toBeCalledTimes(1);
110
+ });
111
+
112
+ test('available languages derrived from staticData', () => {
113
+ const languageGetter = jest.spyOn(window.navigator, 'language', 'get');
114
+ languageGetter.mockReturnValue('cs');
115
+ properties.config = new TolgeeConfig({ staticData: { en: {}, cs: {} } });
116
+ expect(properties.currentLanguage).toEqual('cs');
117
+ });
118
+
119
+ test('returns default lang if not available', () => {
120
+ const languageGetter = jest.spyOn(window.navigator, 'language', 'get');
121
+ languageGetter.mockReturnValue('cs');
122
+ properties.config = new TolgeeConfig({
123
+ availableLanguages: ['en'],
124
+ defaultLanguage: 'en',
125
+ });
126
+ expect(properties.currentLanguage).toEqual('en');
127
+ });
128
+
129
+ test('returns correct language ignoring dialect', () => {
130
+ const languageGetter = jest.spyOn(window.navigator, 'language', 'get');
131
+ languageGetter.mockReturnValue('en-GB');
132
+ properties.config = new TolgeeConfig({
133
+ availableLanguages: ['en-US'],
134
+ defaultLanguage: 'en-US',
135
+ });
136
+ expect(properties.currentLanguage).toEqual('en-US');
137
+ });
138
+
139
+ test('resets current language when missing in availableLanguages', () => {
140
+ const dummyReturn = 'cs';
141
+ Storage.prototype.getItem = jest.fn();
142
+ mocked(localStorage.getItem).mockReturnValueOnce(dummyReturn);
143
+ properties.config = new TolgeeConfig({
144
+ availableLanguages: ['en', 'de'],
145
+ defaultLanguage: 'en',
146
+ });
147
+ expect(properties.currentLanguage).toEqual('en');
148
+ });
149
+ });
150
+ });
@@ -0,0 +1,112 @@
1
+ import { TolgeeConfig } from './TolgeeConfig';
2
+ import { Mode, Scope } from './types';
3
+
4
+ const PREFERRED_LANGUAGES_LOCAL_STORAGE_KEY = '__tolgee_preferredLanguages';
5
+ const CURRENT_LANGUAGE_LOCAL_STORAGE_KEY = '__tolgee_currentLanguage';
6
+
7
+ export class Properties {
8
+ config: TolgeeConfig;
9
+ scopes?: Scope[];
10
+ projectId?: number;
11
+ permittedLanguageIds?: number[];
12
+ mode?: Mode;
13
+ _currentLanguage?: string;
14
+
15
+ get currentLanguage(): string {
16
+ if (this.config?.forceLanguage) {
17
+ return this.config.forceLanguage;
18
+ }
19
+
20
+ if (!this._currentLanguage) {
21
+ this._currentLanguage = this.getInitialLanguage();
22
+ }
23
+
24
+ return this._currentLanguage;
25
+ }
26
+
27
+ set currentLanguage(language: string) {
28
+ if (!language) {
29
+ throw new Error(`Setting invalid language value ${language}`);
30
+ }
31
+ this._currentLanguage = language;
32
+
33
+ if (
34
+ this.config?.enableLanguageStore &&
35
+ typeof localStorage !== 'undefined'
36
+ ) {
37
+ localStorage.setItem(CURRENT_LANGUAGE_LOCAL_STORAGE_KEY, language);
38
+ }
39
+ }
40
+
41
+ get preferredLanguages(): Set<string> {
42
+ return new Set(
43
+ JSON.parse(localStorage.getItem(PREFERRED_LANGUAGES_LOCAL_STORAGE_KEY))
44
+ );
45
+ }
46
+
47
+ set preferredLanguages(languages: Set<string>) {
48
+ localStorage.setItem(
49
+ PREFERRED_LANGUAGES_LOCAL_STORAGE_KEY,
50
+ JSON.stringify(Array.from(languages))
51
+ );
52
+ }
53
+
54
+ private getInitialLanguage() {
55
+ if (this.config.enableLanguageStore) {
56
+ const storedLanguage = this.getStoredLanguage();
57
+ if (storedLanguage) {
58
+ return storedLanguage;
59
+ }
60
+ }
61
+
62
+ if (this.config.enableLanguageDetection) {
63
+ const detectedLanguage = this.getLanguageByNavigator();
64
+ if (detectedLanguage) {
65
+ return detectedLanguage;
66
+ }
67
+ }
68
+
69
+ return this.config.defaultLanguage;
70
+ }
71
+
72
+ private getStoredLanguage() {
73
+ if (typeof localStorage !== 'undefined') {
74
+ const storedLanguage = localStorage.getItem(
75
+ CURRENT_LANGUAGE_LOCAL_STORAGE_KEY
76
+ );
77
+
78
+ if (!this.config.availableLanguages) {
79
+ return storedLanguage;
80
+ }
81
+
82
+ const isSavedLanguageAvailable =
83
+ this.config.availableLanguages.indexOf(storedLanguage) > -1;
84
+
85
+ if (isSavedLanguageAvailable) {
86
+ return storedLanguage;
87
+ }
88
+ }
89
+ }
90
+
91
+ private getLanguageByNavigator() {
92
+ if (typeof window !== 'undefined' && this.config.availableLanguages) {
93
+ const preferred = window.navigator.language;
94
+ const exactMatch = this.config.availableLanguages.find(
95
+ (l) => l === preferred
96
+ );
97
+ if (exactMatch) {
98
+ return exactMatch;
99
+ }
100
+
101
+ const getTwoLetters = (fullTag) => fullTag.replace(/^(.+?)(-.*)?$/, '$1');
102
+
103
+ const preferredTwoLetter = getTwoLetters(window.navigator.language);
104
+ const twoLetterMatch = this.config.availableLanguages.find(
105
+ (l) => getTwoLetters(l) === preferredTwoLetter
106
+ );
107
+ if (twoLetterMatch) {
108
+ return twoLetterMatch;
109
+ }
110
+ }
111
+ }
112
+ }