@tolgee/core 4.9.2 → 4.10.0-rc.14ca700.0

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 (274) hide show
  1. package/dist/tolgee.cjs.js +1075 -7022
  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.min.mjs +1 -1
  6. package/dist/tolgee.esm.min.mjs.map +1 -1
  7. package/dist/tolgee.esm.mjs +1073 -7017
  8. package/dist/tolgee.esm.mjs.map +1 -1
  9. package/dist/tolgee.umd.js +1075 -7022
  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/lib/Controller/Cache/Cache.d.ts +22 -0
  14. package/lib/Controller/Cache/helpers.d.ts +4 -0
  15. package/lib/Controller/Controller.d.ts +137 -0
  16. package/lib/Controller/Events/EventEmitter.d.ts +6 -0
  17. package/lib/Controller/Events/EventEmitterSelective.d.ts +15 -0
  18. package/lib/Controller/Events/Events.d.ts +50 -0
  19. package/lib/Controller/Plugins/Plugins.d.ts +43 -0
  20. package/lib/Controller/State/State.d.ts +21 -0
  21. package/lib/Controller/State/helpers.d.ts +6 -0
  22. package/lib/Controller/State/initState.d.ts +55 -0
  23. package/lib/Controller/ValueObserver.d.ts +5 -0
  24. package/lib/Tolgee.d.ts +2 -68
  25. package/lib/TranslateParams.d.ts +2 -0
  26. package/lib/{Constants/Global.d.ts → constants.d.ts} +1 -2
  27. package/lib/helpers.d.ts +3 -0
  28. package/lib/index.d.ts +3 -8
  29. package/lib/types.d.ts +235 -84
  30. package/package.json +20 -29
  31. package/src/Controller/Cache/Cache.ts +293 -0
  32. package/src/Controller/Cache/helpers.ts +37 -0
  33. package/src/Controller/Controller.ts +307 -0
  34. package/src/Controller/Events/EventEmitter.ts +27 -0
  35. package/src/Controller/Events/EventEmitterSelective.test.ts +125 -0
  36. package/src/Controller/Events/EventEmitterSelective.ts +179 -0
  37. package/src/Controller/Events/Events.ts +66 -0
  38. package/src/Controller/Plugins/Plugins.ts +305 -0
  39. package/src/Controller/State/State.ts +156 -0
  40. package/src/Controller/State/helpers.ts +41 -0
  41. package/src/Controller/State/initState.ts +90 -0
  42. package/src/Controller/ValueObserver.ts +23 -0
  43. package/src/Tolgee.ts +72 -333
  44. package/src/TranslateParams.test.ts +41 -0
  45. package/src/TranslateParams.ts +51 -0
  46. package/src/__test/backend.test.ts +48 -0
  47. package/src/__test/cache.test.ts +148 -0
  48. package/src/__test/client.test.ts +48 -0
  49. package/src/__test/events.test.ts +33 -0
  50. package/src/__test/initialization.test.ts +73 -0
  51. package/src/__test/jest-setup.ts +2 -0
  52. package/src/__test/languageDetection.test.ts +129 -0
  53. package/src/__test/languageStorage.test.ts +145 -0
  54. package/src/__test/languages.test.ts +112 -0
  55. package/src/__test/loading.test.ts +39 -0
  56. package/src/__test/namespaces.test.ts +99 -0
  57. package/src/__test/namespacesFallback.test.ts +74 -0
  58. package/src/__test/plugins.test.ts +98 -0
  59. package/src/__test/testTools.ts +7 -0
  60. package/src/{Constants/Global.ts → constants.ts} +1 -3
  61. package/src/helpers.ts +17 -0
  62. package/src/index.ts +9 -8
  63. package/src/types.ts +322 -90
  64. package/README.md +0 -45
  65. package/dist/Constants/Global.d.ts +0 -6
  66. package/dist/Constants/ModifierKey.d.ts +0 -6
  67. package/dist/Errors/ApiHttpError.d.ts +0 -5
  68. package/dist/Observer.d.ts +0 -14
  69. package/dist/Observer.test.d.ts +0 -2
  70. package/dist/Properties.d.ts +0 -17
  71. package/dist/Properties.test.d.ts +0 -1
  72. package/dist/Tolgee.d.ts +0 -68
  73. package/dist/Tolgee.test.d.ts +0 -1
  74. package/dist/TolgeeConfig.d.ts +0 -69
  75. package/dist/TolgeeConfig.test.d.ts +0 -1
  76. package/dist/__integration/FormatterIcu.test.d.ts +0 -1
  77. package/dist/__integration/FormatterMissing.d.ts +0 -1
  78. package/dist/__integration/Tolgee.test.d.ts +0 -1
  79. package/dist/__integration/TolgeeInvisible.test.d.ts +0 -1
  80. package/dist/__integration/mockTranslations.d.ts +0 -7
  81. package/dist/__integration/testConfig.d.ts +0 -9
  82. package/dist/__testFixtures/classMock.d.ts +0 -3
  83. package/dist/__testFixtures/createElement.d.ts +0 -2
  84. package/dist/__testFixtures/createTestDom.d.ts +0 -9
  85. package/dist/__testFixtures/mocked.d.ts +0 -20
  86. package/dist/__testFixtures/setupAfterEnv.d.ts +0 -8
  87. package/dist/helpers/NodeHelper.d.ts +0 -14
  88. package/dist/helpers/TextHelper.d.ts +0 -5
  89. package/dist/helpers/TextHelper.test.d.ts +0 -1
  90. package/dist/helpers/commonTypes.d.ts +0 -2
  91. package/dist/helpers/encoderPolyfill.d.ts +0 -8
  92. package/dist/helpers/secret.d.ts +0 -6
  93. package/dist/helpers/secret.test.d.ts +0 -1
  94. package/dist/helpers/sleep.d.ts +0 -1
  95. package/dist/highlighter/HighlightFunctionsInitializer.d.ts +0 -10
  96. package/dist/highlighter/HighlightFunctionsInitializer.test.d.ts +0 -1
  97. package/dist/highlighter/MouseEventHandler.d.ts +0 -29
  98. package/dist/highlighter/MouseEventHandler.test.d.ts +0 -1
  99. package/dist/highlighter/TranslationHighlighter.d.ts +0 -14
  100. package/dist/highlighter/TranslationHighlighter.test.d.ts +0 -1
  101. package/dist/index.d.ts +0 -10
  102. package/dist/internal.d.ts +0 -2
  103. package/dist/modules/IcuFormatter.d.ts +0 -2
  104. package/dist/modules/IcuFormatter.test.d.ts +0 -1
  105. package/dist/modules/index.d.ts +0 -1
  106. package/dist/services/ApiHttpService.d.ts +0 -15
  107. package/dist/services/CoreService.d.ts +0 -18
  108. package/dist/services/CoreService.test.d.ts +0 -1
  109. package/dist/services/DependencyService.d.ts +0 -39
  110. package/dist/services/DependencyService.test.d.ts +0 -1
  111. package/dist/services/ElementRegistrar.d.ts +0 -19
  112. package/dist/services/ElementRegistrar.test.d.ts +0 -1
  113. package/dist/services/EventEmitter.d.ts +0 -13
  114. package/dist/services/EventService.d.ts +0 -9
  115. package/dist/services/ModuleService.d.ts +0 -5
  116. package/dist/services/ScreenshotService.d.ts +0 -15
  117. package/dist/services/Subscription.d.ts +0 -5
  118. package/dist/services/TextService.d.ts +0 -14
  119. package/dist/services/TextService.test.d.ts +0 -1
  120. package/dist/services/TranslationService.d.ts +0 -75
  121. package/dist/services/TranslationService.test.d.ts +0 -1
  122. package/dist/services/__mocks__/CoreService.d.ts +0 -2
  123. package/dist/toolsManager/Messages.d.ts +0 -8
  124. package/dist/toolsManager/Messages.test.d.ts +0 -1
  125. package/dist/toolsManager/PluginManager.d.ts +0 -21
  126. package/dist/toolsManager/PluginManager.test.d.ts +0 -1
  127. package/dist/types/DTOs.d.ts +0 -20
  128. package/dist/types/apiSchema.generated.d.ts +0 -6185
  129. package/dist/types.d.ts +0 -123
  130. package/dist/wrappers/AbstractWrapper.d.ts +0 -8
  131. package/dist/wrappers/NodeHandler.d.ts +0 -18
  132. package/dist/wrappers/WrappedHandler.d.ts +0 -8
  133. package/dist/wrappers/invisible/AttributeHandler.d.ts +0 -8
  134. package/dist/wrappers/invisible/Coder.d.ts +0 -7
  135. package/dist/wrappers/invisible/ContentHandler.d.ts +0 -6
  136. package/dist/wrappers/invisible/CoreHandler.d.ts +0 -10
  137. package/dist/wrappers/invisible/InvisibleWrapper.d.ts +0 -18
  138. package/dist/wrappers/invisible/ValueMemory.d.ts +0 -5
  139. package/dist/wrappers/invisible/ValueMemory.test.d.ts +0 -1
  140. package/dist/wrappers/text/AttributeHandler.d.ts +0 -8
  141. package/dist/wrappers/text/AttributeHandler.test.d.ts +0 -1
  142. package/dist/wrappers/text/Coder.d.ts +0 -15
  143. package/dist/wrappers/text/Coder.test.d.ts +0 -1
  144. package/dist/wrappers/text/ContentHandler.d.ts +0 -8
  145. package/dist/wrappers/text/ContentHandler.test.d.ts +0 -1
  146. package/dist/wrappers/text/CoreHandler.d.ts +0 -17
  147. package/dist/wrappers/text/CoreHandler.test.d.ts +0 -1
  148. package/dist/wrappers/text/TextWrapper.d.ts +0 -20
  149. package/index.js +0 -7
  150. package/lib/Constants/ModifierKey.d.ts +0 -6
  151. package/lib/Errors/ApiHttpError.d.ts +0 -5
  152. package/lib/Observer.d.ts +0 -14
  153. package/lib/Properties.d.ts +0 -17
  154. package/lib/TolgeeConfig.d.ts +0 -69
  155. package/lib/helpers/NodeHelper.d.ts +0 -14
  156. package/lib/helpers/TextHelper.d.ts +0 -5
  157. package/lib/helpers/commonTypes.d.ts +0 -2
  158. package/lib/helpers/encoderPolyfill.d.ts +0 -8
  159. package/lib/helpers/secret.d.ts +0 -6
  160. package/lib/helpers/sleep.d.ts +0 -1
  161. package/lib/highlighter/HighlightFunctionsInitializer.d.ts +0 -10
  162. package/lib/highlighter/MouseEventHandler.d.ts +0 -29
  163. package/lib/highlighter/TranslationHighlighter.d.ts +0 -14
  164. package/lib/modules/IcuFormatter.d.ts +0 -2
  165. package/lib/modules/index.d.ts +0 -1
  166. package/lib/services/ApiHttpService.d.ts +0 -15
  167. package/lib/services/CoreService.d.ts +0 -18
  168. package/lib/services/DependencyService.d.ts +0 -39
  169. package/lib/services/ElementRegistrar.d.ts +0 -19
  170. package/lib/services/EventEmitter.d.ts +0 -13
  171. package/lib/services/EventService.d.ts +0 -9
  172. package/lib/services/ModuleService.d.ts +0 -5
  173. package/lib/services/ScreenshotService.d.ts +0 -15
  174. package/lib/services/Subscription.d.ts +0 -5
  175. package/lib/services/TextService.d.ts +0 -14
  176. package/lib/services/TranslationService.d.ts +0 -75
  177. package/lib/toolsManager/Messages.d.ts +0 -8
  178. package/lib/toolsManager/PluginManager.d.ts +0 -21
  179. package/lib/types/DTOs.d.ts +0 -20
  180. package/lib/types/apiSchema.generated.d.ts +0 -6185
  181. package/lib/wrappers/AbstractWrapper.d.ts +0 -8
  182. package/lib/wrappers/NodeHandler.d.ts +0 -18
  183. package/lib/wrappers/WrappedHandler.d.ts +0 -8
  184. package/lib/wrappers/invisible/AttributeHandler.d.ts +0 -8
  185. package/lib/wrappers/invisible/Coder.d.ts +0 -7
  186. package/lib/wrappers/invisible/ContentHandler.d.ts +0 -6
  187. package/lib/wrappers/invisible/CoreHandler.d.ts +0 -10
  188. package/lib/wrappers/invisible/InvisibleWrapper.d.ts +0 -18
  189. package/lib/wrappers/invisible/ValueMemory.d.ts +0 -5
  190. package/lib/wrappers/text/AttributeHandler.d.ts +0 -8
  191. package/lib/wrappers/text/Coder.d.ts +0 -15
  192. package/lib/wrappers/text/ContentHandler.d.ts +0 -8
  193. package/lib/wrappers/text/CoreHandler.d.ts +0 -17
  194. package/lib/wrappers/text/TextWrapper.d.ts +0 -20
  195. package/src/Constants/ModifierKey.ts +0 -6
  196. package/src/Errors/ApiHttpError.ts +0 -8
  197. package/src/Observer.test.ts +0 -119
  198. package/src/Observer.ts +0 -68
  199. package/src/Properties.test.ts +0 -150
  200. package/src/Properties.ts +0 -112
  201. package/src/Tolgee.test.ts +0 -473
  202. package/src/TolgeeConfig.test.ts +0 -21
  203. package/src/TolgeeConfig.ts +0 -134
  204. package/src/__integration/FormatterIcu.test.ts +0 -80
  205. package/src/__integration/FormatterMissing.ts +0 -54
  206. package/src/__integration/Tolgee.test.ts +0 -90
  207. package/src/__integration/TolgeeInvisible.test.ts +0 -145
  208. package/src/__integration/mockTranslations.ts +0 -6
  209. package/src/__integration/testConfig.ts +0 -16
  210. package/src/__testFixtures/classMock.ts +0 -11
  211. package/src/__testFixtures/createElement.ts +0 -43
  212. package/src/__testFixtures/createTestDom.ts +0 -26
  213. package/src/__testFixtures/mocked.ts +0 -25
  214. package/src/__testFixtures/setupAfterEnv.ts +0 -34
  215. package/src/helpers/NodeHelper.ts +0 -90
  216. package/src/helpers/TextHelper.test.ts +0 -62
  217. package/src/helpers/TextHelper.ts +0 -58
  218. package/src/helpers/commonTypes.ts +0 -8
  219. package/src/helpers/encoderPolyfill.ts +0 -96
  220. package/src/helpers/secret.test.ts +0 -61
  221. package/src/helpers/secret.ts +0 -68
  222. package/src/helpers/sleep.ts +0 -2
  223. package/src/highlighter/HighlightFunctionsInitializer.test.ts +0 -40
  224. package/src/highlighter/HighlightFunctionsInitializer.ts +0 -61
  225. package/src/highlighter/MouseEventHandler.test.ts +0 -151
  226. package/src/highlighter/MouseEventHandler.ts +0 -191
  227. package/src/highlighter/TranslationHighlighter.test.ts +0 -177
  228. package/src/highlighter/TranslationHighlighter.ts +0 -113
  229. package/src/internal.ts +0 -2
  230. package/src/modules/IcuFormatter.test.ts +0 -21
  231. package/src/modules/IcuFormatter.ts +0 -39
  232. package/src/modules/index.ts +0 -1
  233. package/src/services/ApiHttpService.ts +0 -85
  234. package/src/services/CoreService.test.ts +0 -141
  235. package/src/services/CoreService.ts +0 -76
  236. package/src/services/DependencyService.test.ts +0 -51
  237. package/src/services/DependencyService.ts +0 -116
  238. package/src/services/ElementRegistrar.test.ts +0 -131
  239. package/src/services/ElementRegistrar.ts +0 -108
  240. package/src/services/EventEmitter.ts +0 -52
  241. package/src/services/EventService.ts +0 -14
  242. package/src/services/ModuleService.ts +0 -14
  243. package/src/services/ScreenshotService.ts +0 -31
  244. package/src/services/Subscription.ts +0 -7
  245. package/src/services/TextService.test.ts +0 -88
  246. package/src/services/TextService.ts +0 -82
  247. package/src/services/TranslationService.test.ts +0 -358
  248. package/src/services/TranslationService.ts +0 -417
  249. package/src/services/__mocks__/CoreService.ts +0 -17
  250. package/src/toolsManager/Messages.test.ts +0 -79
  251. package/src/toolsManager/Messages.ts +0 -60
  252. package/src/toolsManager/PluginManager.test.ts +0 -108
  253. package/src/toolsManager/PluginManager.ts +0 -129
  254. package/src/types/DTOs.ts +0 -25
  255. package/src/types/apiSchema.generated.ts +0 -6208
  256. package/src/wrappers/AbstractWrapper.ts +0 -14
  257. package/src/wrappers/NodeHandler.ts +0 -143
  258. package/src/wrappers/WrappedHandler.ts +0 -28
  259. package/src/wrappers/invisible/AttributeHandler.ts +0 -23
  260. package/src/wrappers/invisible/Coder.ts +0 -65
  261. package/src/wrappers/invisible/ContentHandler.ts +0 -15
  262. package/src/wrappers/invisible/CoreHandler.ts +0 -17
  263. package/src/wrappers/invisible/InvisibleWrapper.ts +0 -59
  264. package/src/wrappers/invisible/ValueMemory.test.ts +0 -25
  265. package/src/wrappers/invisible/ValueMemory.ts +0 -16
  266. package/src/wrappers/text/AttributeHandler.test.ts +0 -118
  267. package/src/wrappers/text/AttributeHandler.ts +0 -25
  268. package/src/wrappers/text/Coder.test.ts +0 -298
  269. package/src/wrappers/text/Coder.ts +0 -202
  270. package/src/wrappers/text/ContentHandler.test.ts +0 -185
  271. package/src/wrappers/text/ContentHandler.ts +0 -21
  272. package/src/wrappers/text/CoreHandler.test.ts +0 -106
  273. package/src/wrappers/text/CoreHandler.ts +0 -45
  274. package/src/wrappers/text/TextWrapper.ts +0 -69
@@ -0,0 +1,293 @@
1
+ import {
2
+ CacheAsyncRequests,
3
+ CacheDescriptor,
4
+ CacheDescriptorInternal,
5
+ CacheDescriptorWithKey,
6
+ EventEmitterType,
7
+ FallbackNSTranslation,
8
+ Options,
9
+ TranslationsFlat,
10
+ TranslationValue,
11
+ TreeTranslationsData,
12
+ BackendGetRecord,
13
+ BackendGetDevRecord,
14
+ } from '../../types';
15
+ import { getFallbackArray } from '../State/helpers';
16
+ import { ValueObserverInstance } from '../ValueObserver';
17
+
18
+ import { decodeCacheKey, encodeCacheKey, flattenTranslations } from './helpers';
19
+
20
+ type CacheRecord = {
21
+ version: number;
22
+ data: TranslationsFlat;
23
+ };
24
+
25
+ type StateCache = Map<string, CacheRecord>;
26
+
27
+ export const Cache = (
28
+ onCacheChange: EventEmitterType<CacheDescriptorWithKey>,
29
+ backendGetRecord: BackendGetRecord,
30
+ backendGetDevRecord: BackendGetDevRecord,
31
+ withDefaultNs: (descriptor: CacheDescriptor) => CacheDescriptorInternal,
32
+ isInitialLoading: () => boolean,
33
+ fetchingObserver: ValueObserverInstance<boolean>,
34
+ loadingObserver: ValueObserverInstance<boolean>
35
+ ) => {
36
+ const asyncRequests: CacheAsyncRequests = new Map();
37
+ const cache: StateCache = new Map();
38
+ let staticData: NonNullable<Options['staticData']> = {};
39
+ let version = 0;
40
+
41
+ function addStaticData(data: Options['staticData']) {
42
+ if (data) {
43
+ staticData = { ...staticData, ...data };
44
+ Object.entries(data).forEach(([key, value]) => {
45
+ if (typeof value !== 'function') {
46
+ const descriptor = decodeCacheKey(key);
47
+ const existing = cache.get(key);
48
+ if (!existing || existing.version === 0) {
49
+ addRecordInternal(descriptor, value, 0);
50
+ }
51
+ }
52
+ });
53
+ }
54
+ }
55
+
56
+ function invalidate() {
57
+ asyncRequests.clear();
58
+ version += 1;
59
+ }
60
+
61
+ function addRecordInternal(
62
+ descriptor: CacheDescriptorInternal,
63
+ data: TreeTranslationsData,
64
+ recordVersion: number
65
+ ) {
66
+ const cacheKey = encodeCacheKey(descriptor);
67
+ cache.set(cacheKey, {
68
+ data: flattenTranslations(data),
69
+ version: recordVersion,
70
+ });
71
+ onCacheChange.emit(descriptor);
72
+ }
73
+
74
+ function addRecord(
75
+ descriptor: CacheDescriptorInternal,
76
+ data: TreeTranslationsData
77
+ ) {
78
+ addRecordInternal(descriptor, data, version);
79
+ }
80
+
81
+ function exists(descriptor: CacheDescriptorInternal, strict = false) {
82
+ const record = cache.get(encodeCacheKey(descriptor));
83
+ if (record && strict) {
84
+ return record.version === version;
85
+ }
86
+ return Boolean(record);
87
+ }
88
+
89
+ function getRecord(descriptor: CacheDescriptor) {
90
+ return cache.get(encodeCacheKey(withDefaultNs(descriptor)))?.data;
91
+ }
92
+
93
+ function getTranslation(descriptor: CacheDescriptorInternal, key: string) {
94
+ return cache.get(encodeCacheKey(descriptor))?.data.get(key);
95
+ }
96
+
97
+ function getTranslationNs(
98
+ namespaces: string[],
99
+ languages: string[],
100
+ key: string
101
+ ) {
102
+ for (const namespace of namespaces) {
103
+ for (const language of languages) {
104
+ const value = cache
105
+ .get(encodeCacheKey({ language, namespace }))
106
+ ?.data.get(key);
107
+ if (value !== undefined && value !== null) {
108
+ return namespace;
109
+ }
110
+ }
111
+ }
112
+ return Array.from(new Set(namespaces));
113
+ }
114
+
115
+ function getTranslationFallback(
116
+ namespaces: string[],
117
+ languages: string[],
118
+ key: string
119
+ ) {
120
+ for (const namespace of namespaces) {
121
+ for (const language of languages) {
122
+ const value = cache
123
+ .get(encodeCacheKey({ language, namespace }))
124
+ ?.data.get(key);
125
+ if (value !== undefined && value !== null) {
126
+ return value;
127
+ }
128
+ }
129
+ }
130
+ return undefined;
131
+ }
132
+
133
+ function changeTranslation(
134
+ descriptor: CacheDescriptorInternal,
135
+ key: string,
136
+ value: TranslationValue
137
+ ) {
138
+ const record = cache.get(encodeCacheKey(descriptor))?.data;
139
+ record?.set(key, value);
140
+ onCacheChange.emit({ ...descriptor, key });
141
+ }
142
+
143
+ function isFetching(ns?: FallbackNSTranslation) {
144
+ if (isInitialLoading()) {
145
+ return true;
146
+ }
147
+
148
+ if (ns === undefined) {
149
+ return asyncRequests.size > 0;
150
+ }
151
+ const namespaces = getFallbackArray(ns);
152
+ return Boolean(
153
+ Array.from(asyncRequests.keys()).find((key) =>
154
+ namespaces.includes(decodeCacheKey(key).namespace)
155
+ )
156
+ );
157
+ }
158
+
159
+ function isLoading(language: string | undefined, ns?: FallbackNSTranslation) {
160
+ const namespaces = getFallbackArray(ns);
161
+
162
+ return Boolean(
163
+ isInitialLoading() ||
164
+ Array.from(asyncRequests.keys()).find((key) => {
165
+ const descriptor = decodeCacheKey(key);
166
+ return (
167
+ (!namespaces.length || namespaces.includes(descriptor.namespace)) &&
168
+ !exists({
169
+ namespace: descriptor.namespace,
170
+ language: language!,
171
+ })
172
+ );
173
+ })
174
+ );
175
+ }
176
+
177
+ function fetchNormal(keyObject: CacheDescriptorInternal) {
178
+ let dataPromise = undefined as
179
+ | Promise<TreeTranslationsData | undefined>
180
+ | undefined;
181
+ if (!dataPromise) {
182
+ const staticDataValue = staticData[encodeCacheKey(keyObject)];
183
+ if (typeof staticDataValue === 'function') {
184
+ dataPromise = staticDataValue();
185
+ }
186
+ }
187
+
188
+ if (!dataPromise) {
189
+ dataPromise = backendGetRecord(keyObject);
190
+ }
191
+
192
+ if (!dataPromise) {
193
+ // return empty data, so we know it has already been attempted to fetch
194
+ dataPromise = Promise.resolve({});
195
+ }
196
+ return dataPromise;
197
+ }
198
+
199
+ function fetchData(keyObject: CacheDescriptorInternal, isDev: boolean) {
200
+ let dataPromise = undefined as
201
+ | Promise<TreeTranslationsData | undefined>
202
+ | undefined;
203
+ if (isDev) {
204
+ dataPromise = backendGetDevRecord(keyObject)?.catch(() => {
205
+ // eslint-disable-next-line no-console
206
+ console.warn(`Tolgee: Failed to fetch data from dev backend`);
207
+ // fallback to normal fetch if dev fails
208
+ return fetchNormal(keyObject);
209
+ });
210
+ }
211
+
212
+ if (!dataPromise) {
213
+ dataPromise = fetchNormal(keyObject);
214
+ }
215
+
216
+ return dataPromise;
217
+ }
218
+
219
+ async function loadRecords(descriptors: CacheDescriptor[], isDev: boolean) {
220
+ const withPromises = descriptors.map((descriptor) => {
221
+ const keyObject = withDefaultNs(descriptor);
222
+ const cacheKey = encodeCacheKey(keyObject);
223
+ const existingPromise = asyncRequests.get(cacheKey);
224
+
225
+ if (existingPromise) {
226
+ return {
227
+ new: false,
228
+ promise: existingPromise,
229
+ keyObject,
230
+ cacheKey,
231
+ };
232
+ }
233
+ const dataPromise = fetchData(keyObject, isDev);
234
+ asyncRequests.set(cacheKey, dataPromise);
235
+ return {
236
+ new: true,
237
+ promise: dataPromise,
238
+ keyObject,
239
+ cacheKey,
240
+ };
241
+ });
242
+ fetchingObserver.notify();
243
+ loadingObserver.notify();
244
+
245
+ const results = await Promise.all(withPromises.map((val) => val.promise));
246
+
247
+ withPromises.forEach((value, i) => {
248
+ const promiseChanged =
249
+ asyncRequests.get(value.cacheKey) !== value.promise;
250
+ // if promise has changed in between, it means cache been invalidated or
251
+ // new data are being fetched
252
+ if (value.new && !promiseChanged) {
253
+ asyncRequests.delete(value.cacheKey);
254
+ const data = results[i];
255
+ if (data) {
256
+ addRecord(value.keyObject, data);
257
+ }
258
+ }
259
+ });
260
+ fetchingObserver.notify();
261
+ loadingObserver.notify();
262
+
263
+ return withPromises.map((val) => getRecord(val.keyObject)!);
264
+ }
265
+
266
+ function getAllRecords() {
267
+ const entries = Array.from(cache.entries());
268
+ return entries.map(([key, entry]) => {
269
+ return {
270
+ ...decodeCacheKey(key),
271
+ data: entry.data,
272
+ };
273
+ });
274
+ }
275
+
276
+ return Object.freeze({
277
+ addStaticData,
278
+ invalidate,
279
+ addRecord,
280
+ exists,
281
+ getRecord,
282
+ getTranslation,
283
+ getTranslationNs,
284
+ getTranslationFallback,
285
+ changeTranslation,
286
+ isFetching,
287
+ isLoading,
288
+ loadRecords,
289
+ getAllRecords,
290
+ });
291
+ };
292
+
293
+ export type CacheType = ReturnType<typeof Cache>;
@@ -0,0 +1,37 @@
1
+ import { CacheDescriptorInternal, TreeTranslationsData } from '../../types';
2
+
3
+ export const flattenTranslations = (
4
+ data: TreeTranslationsData
5
+ ): Map<string, string> => {
6
+ const result: Map<string, string> = new Map();
7
+ Object.entries(data).forEach(([key, value]) => {
8
+ // ignore empty values
9
+ if (value === undefined || value === null) {
10
+ return;
11
+ }
12
+ if (typeof value === 'object') {
13
+ flattenTranslations(value).forEach((flatValue, flatKey) => {
14
+ result.set(key + '.' + flatKey, flatValue);
15
+ });
16
+ return;
17
+ }
18
+ result.set(key, value as string);
19
+ });
20
+ return result;
21
+ };
22
+
23
+ export const decodeCacheKey = (key: string): CacheDescriptorInternal => {
24
+ const [firstPart, secondPart] = key.split(':');
25
+ return { language: firstPart, namespace: secondPart || '' };
26
+ };
27
+
28
+ export const encodeCacheKey = ({
29
+ language,
30
+ namespace,
31
+ }: CacheDescriptorInternal) => {
32
+ if (namespace) {
33
+ return `${language}:${namespace}`;
34
+ } else {
35
+ return language;
36
+ }
37
+ };
@@ -0,0 +1,307 @@
1
+ import { Events } from './Events/Events';
2
+ import {
3
+ CacheDescriptor,
4
+ FallbackNSTranslation,
5
+ Options,
6
+ TFnType,
7
+ TranslatePropsInternal,
8
+ } from '../types';
9
+ import { Cache } from './Cache/Cache';
10
+ import { getFallbackArray } from './State/helpers';
11
+ import { PluginService } from './Plugins/Plugins';
12
+ import { ValueObserver } from './ValueObserver';
13
+ import { State } from './State/State';
14
+ import { isPromise, missingOptionError, valueOrPromise } from '../helpers';
15
+ import { getTranslateParams } from '../TranslateParams';
16
+
17
+ type StateServiceProps = {
18
+ options?: Partial<Options>;
19
+ };
20
+
21
+ export const Controller = ({ options }: StateServiceProps) => {
22
+ const events = Events(getFallbackNamespaces);
23
+ const fetchingObserver = ValueObserver<boolean>(
24
+ false,
25
+ () => cache.isFetching(),
26
+ events.onFetchingChange.emit
27
+ );
28
+ const loadingObserver = ValueObserver<boolean>(
29
+ false,
30
+ () => isLoading(),
31
+ events.onLoadingChange.emit
32
+ );
33
+
34
+ const state = State(
35
+ events.onLanguageChange,
36
+ events.onPendingLanguageChange,
37
+ events.onRunningChange
38
+ );
39
+
40
+ const pluginService = PluginService(
41
+ state.getLanguage,
42
+ state.getInitialOptions,
43
+ state.getAvailableLanguages,
44
+ getTranslationNs,
45
+ getTranslation,
46
+ changeTranslation
47
+ );
48
+
49
+ const cache = Cache(
50
+ events.onCacheChange,
51
+ pluginService.getBackendRecord,
52
+ pluginService.getBackendDevRecord,
53
+ state.withDefaultNs,
54
+ state.isInitialLoading,
55
+ fetchingObserver,
56
+ loadingObserver
57
+ );
58
+
59
+ if (options) {
60
+ init(options);
61
+ }
62
+
63
+ events.onKeyUpdate.listen(() => {
64
+ if (state.isRunning()) {
65
+ pluginService.retranslate();
66
+ }
67
+ });
68
+
69
+ function changeTranslation(
70
+ descriptor: CacheDescriptor,
71
+ key: string,
72
+ value: string
73
+ ) {
74
+ const keyObject = state.withDefaultNs(descriptor);
75
+ const previousValue = cache.getTranslation(keyObject, key);
76
+ cache.changeTranslation(keyObject, key, value);
77
+ return {
78
+ revert: () => {
79
+ cache.changeTranslation(keyObject, key, previousValue);
80
+ },
81
+ };
82
+ }
83
+
84
+ function getFallbackNamespaces() {
85
+ return state.getFallbackNamespaces();
86
+ }
87
+
88
+ function init(options: Partial<Options>) {
89
+ state.init(options);
90
+ cache.addStaticData(state.getInitialOptions().staticData);
91
+ }
92
+
93
+ function isLoading(ns?: FallbackNSTranslation) {
94
+ return cache.isLoading(state.getLanguage()!, ns);
95
+ }
96
+
97
+ function isDev() {
98
+ return Boolean(
99
+ state.getInitialOptions().apiKey && pluginService.getDevBackend()
100
+ );
101
+ }
102
+
103
+ async function addActiveNs(ns: FallbackNSTranslation, forget?: boolean) {
104
+ if (!forget) {
105
+ state.addActiveNs(ns);
106
+ }
107
+ if (state.isRunning()) {
108
+ await loadRequiredRecords(undefined, ns);
109
+ }
110
+ }
111
+
112
+ function getRequiredRecords(lang?: string, ns?: FallbackNSTranslation) {
113
+ const languages = state.getFallbackLangs(lang);
114
+ const namespaces =
115
+ ns !== undefined ? getFallbackArray(ns) : state.getRequiredNamespaces();
116
+ const result: CacheDescriptor[] = [];
117
+ languages.forEach((language) => {
118
+ namespaces.forEach((namespace) => {
119
+ if (!cache.exists({ language, namespace }, true)) {
120
+ result.push({ language, namespace });
121
+ }
122
+ });
123
+ });
124
+ return result;
125
+ }
126
+
127
+ function isLoaded(ns?: FallbackNSTranslation) {
128
+ const language = state.getLanguage();
129
+ if (!language) {
130
+ return false;
131
+ }
132
+ const languages = state.getFallbackLangs(language);
133
+ const namespaces =
134
+ ns !== undefined ? getFallbackArray(ns) : state.getRequiredNamespaces();
135
+ const result: CacheDescriptor[] = [];
136
+ languages.forEach((language) => {
137
+ namespaces.forEach((namespace) => {
138
+ if (!cache.exists({ language, namespace })) {
139
+ result.push({ language, namespace });
140
+ }
141
+ });
142
+ });
143
+ return result.length === 0;
144
+ }
145
+
146
+ function loadRequiredRecords(lang?: string, ns?: FallbackNSTranslation) {
147
+ const descriptors = getRequiredRecords(lang, ns);
148
+ if (descriptors.length) {
149
+ return valueOrPromise(loadRecords(descriptors), () => {});
150
+ }
151
+ }
152
+
153
+ async function changeLanguage(language: string) {
154
+ if (
155
+ state.getPendingLanguage() === language &&
156
+ state.getLanguage() === language
157
+ ) {
158
+ return;
159
+ }
160
+ state.setPendingLanguage(language);
161
+
162
+ if (state.isRunning()) {
163
+ await loadRequiredRecords(language);
164
+ }
165
+
166
+ if (language === state.getPendingLanguage()) {
167
+ // there might be parallel language change
168
+ // we only want to apply latest
169
+ state.setLanguage(language);
170
+ pluginService.setStoredLanguage(language);
171
+ }
172
+ }
173
+
174
+ function getTranslationNs({
175
+ key,
176
+ ns,
177
+ }: Pick<TranslatePropsInternal, 'key' | 'ns'>) {
178
+ const namespaces =
179
+ ns !== undefined ? getFallbackArray(ns) : state.getFallbackNamespaces();
180
+ const languages = state.getFallbackLangs();
181
+ return cache.getTranslationNs(namespaces, languages, key);
182
+ }
183
+
184
+ function getTranslation({
185
+ key,
186
+ ns,
187
+ }: Pick<TranslatePropsInternal, 'key' | 'ns'>) {
188
+ const namespaces =
189
+ ns !== undefined ? getFallbackArray(ns) : state.getFallbackNamespaces();
190
+ const languages = state.getFallbackLangs();
191
+ return cache.getTranslationFallback(namespaces, languages, key);
192
+ }
193
+
194
+ function loadInitial() {
195
+ const data = valueOrPromise(initializeLanguage(), () => {
196
+ // fail if there is no language
197
+ return loadRequiredRecords();
198
+ });
199
+
200
+ if (isPromise(data)) {
201
+ state.setInitialLoading(true);
202
+ fetchingObserver.notify();
203
+ loadingObserver.notify();
204
+ return Promise.resolve(data).then(() => {
205
+ state.setInitialLoading(false);
206
+ fetchingObserver.notify();
207
+ loadingObserver.notify();
208
+ events.onInitialLoaded.emit();
209
+ });
210
+ } else {
211
+ events.onInitialLoaded.emit();
212
+ }
213
+ }
214
+
215
+ function initializeLanguage() {
216
+ const existingLanguage = state.getLanguage();
217
+ if (existingLanguage) {
218
+ return;
219
+ }
220
+ if (!state.getInitialOptions().defaultLanguage) {
221
+ throw new Error(missingOptionError('defaultLanguage'));
222
+ }
223
+ const languageOrPromise = pluginService.getInitialLanguage();
224
+ return valueOrPromise(languageOrPromise, (lang) => {
225
+ const language =
226
+ (lang as string | undefined) ||
227
+ state.getInitialOptions().defaultLanguage;
228
+ language && state.setLanguage(language);
229
+ });
230
+ }
231
+
232
+ async function loadRecord(descriptor: CacheDescriptor) {
233
+ return (await loadRecords([descriptor]))[0];
234
+ }
235
+
236
+ function loadRecords(descriptors: CacheDescriptor[]) {
237
+ return cache.loadRecords(descriptors, isDev());
238
+ }
239
+
240
+ const checkCorrectConfiguration = () => {
241
+ const languageDetector = pluginService.getLanguageDetector();
242
+ if (languageDetector) {
243
+ const availableLanguages = state.getAvailableLanguages();
244
+ if (!availableLanguages) {
245
+ throw new Error(missingOptionError('availableLanguages'));
246
+ }
247
+ }
248
+ if (!state.getLanguage() && !state.getInitialOptions().defaultLanguage) {
249
+ if (languageDetector) {
250
+ throw new Error(missingOptionError('defaultLanguage'));
251
+ } else {
252
+ throw new Error(missingOptionError('language'));
253
+ }
254
+ }
255
+ };
256
+
257
+ function run() {
258
+ let result: Promise<void> | undefined = undefined;
259
+ checkCorrectConfiguration();
260
+ if (!state.isRunning()) {
261
+ if (isDev()) {
262
+ cache.invalidate();
263
+ }
264
+ state.setRunning(true);
265
+ pluginService.run();
266
+ result = loadInitial();
267
+ }
268
+ return Promise.resolve(result);
269
+ }
270
+
271
+ function stop() {
272
+ if (state.isRunning()) {
273
+ pluginService.stop();
274
+ state.setRunning(false);
275
+ }
276
+ }
277
+
278
+ const t: TFnType = (...args) => {
279
+ // @ts-ignore
280
+ const params = getTranslateParams(...args);
281
+ const translation = getTranslation(params);
282
+ return pluginService.formatTranslation({ ...params, translation });
283
+ };
284
+
285
+ return Object.freeze({
286
+ ...events,
287
+ ...state,
288
+ ...pluginService,
289
+ ...cache,
290
+ init,
291
+ changeLanguage,
292
+ getTranslation,
293
+ changeTranslation,
294
+ addActiveNs,
295
+ loadRequiredRecords,
296
+ loadRecords,
297
+ loadRecord,
298
+ isLoading,
299
+ isLoaded,
300
+ t,
301
+ isDev,
302
+ run,
303
+ stop,
304
+ });
305
+ };
306
+
307
+ export type StateServiceType = ReturnType<typeof Controller>;
@@ -0,0 +1,27 @@
1
+ import { Listener, ListenerHandler } from '../../types';
2
+
3
+ export const EventEmitter = <T>() => {
4
+ let handlers: ListenerHandler<T>[] = [];
5
+
6
+ const listen = (handler: ListenerHandler<T>): Listener => {
7
+ const handlerWrapper: ListenerHandler<T> = (e) => {
8
+ handler(e);
9
+ };
10
+
11
+ handlers.push(handlerWrapper);
12
+
13
+ return {
14
+ unsubscribe: () => {
15
+ handlers = handlers.filter((i) => handlerWrapper !== i);
16
+ },
17
+ };
18
+ };
19
+
20
+ const emit = (data: T) => {
21
+ handlers.forEach((handler) => handler({ value: data }));
22
+ };
23
+
24
+ return Object.freeze({ listen, emit });
25
+ };
26
+
27
+ export type EventEmitterType<T> = ReturnType<typeof EventEmitter<T>>;