@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.
- package/dist/tolgee.cjs.js +2 -4
- package/dist/tolgee.cjs.js.map +1 -1
- package/dist/tolgee.cjs.min.js +1 -1
- package/dist/tolgee.cjs.min.js.map +1 -1
- package/dist/{tolgee.esm.js → tolgee.esm.min.mjs} +2 -2
- package/dist/tolgee.esm.min.mjs.map +1 -0
- package/dist/tolgee.esm.mjs +5690 -0
- package/dist/tolgee.esm.mjs.map +1 -0
- package/dist/tolgee.umd.js +2 -4
- package/dist/tolgee.umd.js.map +1 -1
- package/dist/tolgee.umd.min.js +1 -1
- package/dist/tolgee.umd.min.js.map +1 -1
- package/package.json +10 -9
- package/src/Constants/Global.ts +9 -0
- package/src/Constants/ModifierKey.ts +6 -0
- package/src/Errors/ApiHttpError.ts +8 -0
- package/src/Observer.test.ts +119 -0
- package/src/Observer.ts +68 -0
- package/src/Properties.test.ts +150 -0
- package/src/Properties.ts +112 -0
- package/src/Tolgee.test.ts +473 -0
- package/src/Tolgee.ts +335 -0
- package/src/TolgeeConfig.test.ts +21 -0
- package/src/TolgeeConfig.ts +134 -0
- package/src/__integration/FormatterIcu.test.ts +80 -0
- package/src/__integration/FormatterMissing.ts +54 -0
- package/src/__integration/Tolgee.test.ts +90 -0
- package/src/__integration/TolgeeInvisible.test.ts +145 -0
- package/src/__integration/mockTranslations.ts +6 -0
- package/src/__integration/testConfig.ts +16 -0
- package/src/__testFixtures/classMock.ts +11 -0
- package/src/__testFixtures/createElement.ts +43 -0
- package/src/__testFixtures/createTestDom.ts +25 -0
- package/src/__testFixtures/mocked.ts +25 -0
- package/src/__testFixtures/setupAfterEnv.ts +34 -0
- package/src/helpers/NodeHelper.ts +90 -0
- package/src/helpers/TextHelper.test.ts +62 -0
- package/src/helpers/TextHelper.ts +58 -0
- package/src/helpers/commonTypes.ts +8 -0
- package/src/helpers/encoderPolyfill.ts +96 -0
- package/src/helpers/secret.test.ts +61 -0
- package/src/helpers/secret.ts +68 -0
- package/src/helpers/sleep.ts +2 -0
- package/src/highlighter/HighlightFunctionsInitializer.test.ts +40 -0
- package/src/highlighter/HighlightFunctionsInitializer.ts +61 -0
- package/src/highlighter/MouseEventHandler.test.ts +151 -0
- package/src/highlighter/MouseEventHandler.ts +191 -0
- package/src/highlighter/TranslationHighlighter.test.ts +177 -0
- package/src/highlighter/TranslationHighlighter.ts +113 -0
- package/src/index.ts +10 -0
- package/src/internal.ts +2 -0
- package/src/modules/IcuFormatter.ts +17 -0
- package/src/modules/index.ts +1 -0
- package/src/services/ApiHttpService.ts +85 -0
- package/src/services/CoreService.test.ts +142 -0
- package/src/services/CoreService.ts +76 -0
- package/src/services/DependencyService.test.ts +51 -0
- package/src/services/DependencyService.ts +116 -0
- package/src/services/ElementRegistrar.test.ts +131 -0
- package/src/services/ElementRegistrar.ts +108 -0
- package/src/services/EventEmitter.ts +52 -0
- package/src/services/EventService.ts +14 -0
- package/src/services/ModuleService.ts +14 -0
- package/src/services/ScreenshotService.ts +31 -0
- package/src/services/Subscription.ts +7 -0
- package/src/services/TextService.test.ts +88 -0
- package/src/services/TextService.ts +82 -0
- package/src/services/TranslationService.test.ts +358 -0
- package/src/services/TranslationService.ts +417 -0
- package/src/services/__mocks__/CoreService.ts +17 -0
- package/src/toolsManager/Messages.test.ts +79 -0
- package/src/toolsManager/Messages.ts +60 -0
- package/src/toolsManager/PluginManager.test.ts +108 -0
- package/src/toolsManager/PluginManager.ts +129 -0
- package/src/types/DTOs.ts +25 -0
- package/src/types/apiSchema.generated.ts +6208 -0
- package/src/types.ts +146 -0
- package/src/wrappers/AbstractWrapper.ts +14 -0
- package/src/wrappers/NodeHandler.ts +143 -0
- package/src/wrappers/WrappedHandler.ts +28 -0
- package/src/wrappers/invisible/AttributeHandler.ts +23 -0
- package/src/wrappers/invisible/Coder.ts +65 -0
- package/src/wrappers/invisible/ContentHandler.ts +15 -0
- package/src/wrappers/invisible/CoreHandler.ts +17 -0
- package/src/wrappers/invisible/InvisibleWrapper.ts +59 -0
- package/src/wrappers/invisible/ValueMemory.test.ts +25 -0
- package/src/wrappers/invisible/ValueMemory.ts +16 -0
- package/src/wrappers/text/AttributeHandler.test.ts +117 -0
- package/src/wrappers/text/AttributeHandler.ts +25 -0
- package/src/wrappers/text/Coder.test.ts +298 -0
- package/src/wrappers/text/Coder.ts +202 -0
- package/src/wrappers/text/ContentHandler.test.ts +185 -0
- package/src/wrappers/text/ContentHandler.ts +21 -0
- package/src/wrappers/text/CoreHandler.test.ts +106 -0
- package/src/wrappers/text/CoreHandler.ts +45 -0
- package/src/wrappers/text/TextWrapper.ts +69 -0
- 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.
|
|
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.
|
|
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.
|
|
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.
|
|
60
|
+
"tslib": "^2.4.0",
|
|
60
61
|
"typedoc": "0.22.15",
|
|
61
|
-
"typedoc-plugin-markdown": "3.
|
|
62
|
-
"typescript": "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.
|
|
67
|
+
"import": "./dist/tolgee.esm.mjs"
|
|
67
68
|
},
|
|
68
|
-
"gitHead": "
|
|
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,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
|
+
});
|
package/src/Observer.ts
ADDED
|
@@ -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
|
+
}
|