@wavemaker/app-ng-runtime 11.14.1-8.6337 → 11.14.1-9.6343
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/components/base/bundles/index.umd.js +8 -77
- package/components/base/esm2022/pipes/custom-pipes.mjs +1 -1
- package/components/base/esm2022/widgets/common/base/base.component.mjs +7 -67
- package/components/base/esm2022/widgets/common/lazy-load/lazy-load.directive.mjs +3 -7
- package/components/base/esm2022/widgets/framework/property-change-handler.mjs +2 -7
- package/components/base/fesm2022/index.mjs +9 -78
- package/components/base/fesm2022/index.mjs.map +1 -1
- package/components/basic/label/bundles/index.umd.js +1 -9
- package/components/basic/label/esm2022/label.directive.mjs +2 -10
- package/components/basic/label/fesm2022/index.mjs +1 -9
- package/components/basic/label/fesm2022/index.mjs.map +1 -1
- package/components/data/pagination/bundles/index.umd.js +0 -4
- package/components/data/pagination/esm2022/pagination.component.mjs +1 -5
- package/components/data/pagination/fesm2022/index.mjs +0 -4
- package/components/data/pagination/fesm2022/index.mjs.map +1 -1
- package/components/data/table/bundles/index.umd.js +12 -219
- package/components/data/table/esm2022/table-cud.directive.mjs +2 -2
- package/components/data/table/esm2022/table.component.mjs +12 -219
- package/components/data/table/fesm2022/index.mjs +12 -219
- package/components/data/table/fesm2022/index.mjs.map +1 -1
- package/components/data/table/table.component.d.ts +2 -6
- package/components/navigation/menu/bundles/index.umd.js +0 -5
- package/components/navigation/menu/esm2022/menu.component.mjs +1 -6
- package/components/navigation/menu/fesm2022/index.mjs +0 -5
- package/components/navigation/menu/fesm2022/index.mjs.map +1 -1
- package/components/navigation/popover/bundles/index.umd.js +6 -6
- package/components/navigation/popover/esm2022/popover.component.mjs +4 -4
- package/components/navigation/popover/fesm2022/index.mjs +3 -3
- package/components/navigation/popover/fesm2022/index.mjs.map +1 -1
- package/components/navigation/popover/popover.component.d.ts +0 -6
- package/core/bundles/index.umd.js +82 -395
- package/core/esm2022/public_api.mjs +2 -2
- package/core/esm2022/utils/watcher.mjs +81 -392
- package/core/fesm2022/index.mjs +84 -395
- package/core/fesm2022/index.mjs.map +1 -1
- package/core/public_api.d.ts +1 -1
- package/core/utils/watcher.d.ts +5 -28
- package/npm-shrinkwrap.json +2 -2
- package/package-lock.json +2 -2
- package/package.json +1 -1
- package/runtime/base/bundles/index.umd.js +0 -20
- package/runtime/base/esm2022/components/app-component/app.component.mjs +2 -4
- package/runtime/base/esm2022/components/base-page.component.mjs +2 -6
- package/runtime/base/esm2022/components/base-partial.component.mjs +2 -7
- package/runtime/base/esm2022/components/base-prefab.component.mjs +2 -7
- package/runtime/base/esm2022/components/base-spa-page.component.mjs +2 -6
- package/runtime/base/fesm2022/index.mjs +1 -21
- package/runtime/base/fesm2022/index.mjs.map +1 -1
- package/scripts/datatable/datatable.js +2 -34
|
@@ -2,89 +2,14 @@ import { IDGenerator } from './id-generator';
|
|
|
2
2
|
import { $parseExpr } from './expression-parser';
|
|
3
3
|
import { findValueOf } from './utils';
|
|
4
4
|
import { clone, flatten, isArray, isEqual, isObject } from "lodash-es";
|
|
5
|
-
// Constants
|
|
6
|
-
const WIDGET_ID_REGEX = /^(widget-[^_]+)/;
|
|
7
|
-
const WIDGET_PROPERTY_REGEX = /^widget-[^_]+_(.+)$/;
|
|
8
|
-
const ARRAY_INDEX_PLACEHOLDER = '[$i]';
|
|
9
|
-
const ARRAY_INDEX_ZERO = '[0]';
|
|
10
|
-
const DEBOUNCE_WAIT = 100;
|
|
11
|
-
const MAX_WATCH_CYCLES = 5;
|
|
12
5
|
const registry = new Map();
|
|
13
6
|
const watchIdGenerator = new IDGenerator('watch-id-');
|
|
14
|
-
export const FIRST_TIME_WATCH =
|
|
15
|
-
|
|
7
|
+
export const FIRST_TIME_WATCH = {};
|
|
8
|
+
Object.freeze(FIRST_TIME_WATCH);
|
|
9
|
+
export const isFirstTimeChange = v => v === FIRST_TIME_WATCH;
|
|
16
10
|
let muted = false;
|
|
17
|
-
let changedByWatch = false;
|
|
18
|
-
let skipWatchers = false;
|
|
19
|
-
let ngZone;
|
|
20
11
|
let appRef;
|
|
21
|
-
|
|
22
|
-
* CLEANUP SCHEDULER WITH:
|
|
23
|
-
* - Significant watcher delta trigger (+/-)
|
|
24
|
-
* - UI stabilization delay (1–2 seconds)
|
|
25
|
-
* - Cooldown to prevent repeated scheduling
|
|
26
|
-
********************************************************************/
|
|
27
|
-
const CLEANUP_TRIGGER_DELTA = 300; // watcher change threshold
|
|
28
|
-
const CLEANUP_DELAY = 1500; // delay before running cleanup
|
|
29
|
-
const CLEANUP_COOLDOWN = 4000; // prevent re-scheduling for 4s
|
|
30
|
-
let lastWatcherCount = 0;
|
|
31
|
-
let scheduledCleanup = null;
|
|
32
|
-
let lastScheduledTime = 0;
|
|
33
|
-
function getWatcherCount() {
|
|
34
|
-
let count = 0;
|
|
35
|
-
registry.forEach(bucket => count += bucket.size);
|
|
36
|
-
return count;
|
|
37
|
-
}
|
|
38
|
-
export const cleanupStaleWatchers = () => {
|
|
39
|
-
// console.log(".........Cleaning up stale watchers...registry.size....", registry.size);
|
|
40
|
-
let removed = 0;
|
|
41
|
-
registry.forEach((bucket, widgetId) => {
|
|
42
|
-
if (!document.querySelector(`[widget-id="${widgetId}"]`)) {
|
|
43
|
-
for (const key in bucket) {
|
|
44
|
-
if (bucket.hasOwnProperty(key) && key !== "scopeType" && key !== "scopeName" && typeof bucket[key] !== "function") {
|
|
45
|
-
let watchInfo = bucket[key];
|
|
46
|
-
if (watchInfo && watchInfo.destroyFn) {
|
|
47
|
-
watchInfo.destroyFn();
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
}
|
|
51
|
-
removed++;
|
|
52
|
-
registry.delete(widgetId);
|
|
53
|
-
return;
|
|
54
|
-
}
|
|
55
|
-
});
|
|
56
|
-
return removed;
|
|
57
|
-
};
|
|
58
|
-
export const scheduleThresholdCleanup = () => {
|
|
59
|
-
const now = performance.now();
|
|
60
|
-
let lastTriggerPeriod = now - lastScheduledTime;
|
|
61
|
-
// If a cleanup was scheduled recently, skip scheduling
|
|
62
|
-
if (lastTriggerPeriod < CLEANUP_COOLDOWN) {
|
|
63
|
-
return;
|
|
64
|
-
}
|
|
65
|
-
const current = getWatcherCount();
|
|
66
|
-
const delta = Math.abs(current - lastWatcherCount); // significant + or -
|
|
67
|
-
// If change not large enough, skip scheduling
|
|
68
|
-
if (delta <= CLEANUP_TRIGGER_DELTA) {
|
|
69
|
-
lastWatcherCount = current;
|
|
70
|
-
return;
|
|
71
|
-
}
|
|
72
|
-
// Prevent re-scheduling: set timestamp early
|
|
73
|
-
lastScheduledTime = now;
|
|
74
|
-
// Clear previous scheduled cleanup (if any)
|
|
75
|
-
if (scheduledCleanup) {
|
|
76
|
-
clearTimeout(scheduledCleanup);
|
|
77
|
-
}
|
|
78
|
-
// Schedule cleanup after UI stabilizes (delay prevents aggressive cleanup)
|
|
79
|
-
scheduledCleanup = setTimeout(() => {
|
|
80
|
-
cleanupStaleWatchers();
|
|
81
|
-
scheduledCleanup = null;
|
|
82
|
-
}, CLEANUP_DELAY);
|
|
83
|
-
lastWatcherCount = current;
|
|
84
|
-
};
|
|
85
|
-
// Utility functions
|
|
86
|
-
export const isFirstTimeChange = (v) => v === FIRST_TIME_WATCH;
|
|
87
|
-
export const debounce = (fn, wait = DEBOUNCE_WAIT) => {
|
|
12
|
+
export const debounce = (fn, wait = 50) => {
|
|
88
13
|
let timeout;
|
|
89
14
|
return (...args) => {
|
|
90
15
|
window['__zone_symbol__clearTimeout'](timeout);
|
|
@@ -98,350 +23,115 @@ export const unMuteWatchers = () => {
|
|
|
98
23
|
muted = false;
|
|
99
24
|
triggerWatchers();
|
|
100
25
|
};
|
|
101
|
-
/**
|
|
102
|
-
* Extracts widget ID from identifier (e.g., "widget-id23_eventsource" -> "widget-id23")
|
|
103
|
-
*/
|
|
104
|
-
const getWidgetId = (identifier) => {
|
|
105
|
-
if (!identifier || typeof identifier !== 'string') {
|
|
106
|
-
return null;
|
|
107
|
-
}
|
|
108
|
-
const match = identifier.match(WIDGET_ID_REGEX);
|
|
109
|
-
return match ? match[1] : null;
|
|
110
|
-
};
|
|
111
|
-
/**
|
|
112
|
-
* Extracts property name from identifier (e.g., "widget-id23_eventsource" -> "eventsource")
|
|
113
|
-
*/
|
|
114
|
-
const getPropertyName = (identifier) => {
|
|
115
|
-
if (!identifier || typeof identifier !== 'string') {
|
|
116
|
-
return identifier;
|
|
117
|
-
}
|
|
118
|
-
const match = identifier.match(WIDGET_PROPERTY_REGEX);
|
|
119
|
-
return match ? match[1] : identifier;
|
|
120
|
-
};
|
|
121
|
-
/**
|
|
122
|
-
* Array consumer wrapper for array-based expressions
|
|
123
|
-
*/
|
|
124
26
|
const arrayConsumer = (listenerFn, restExpr, newVal, oldVal) => {
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
27
|
+
let data = newVal, formattedData;
|
|
28
|
+
if (isArray(data)) {
|
|
29
|
+
formattedData = data.map(function (datum) {
|
|
30
|
+
return findValueOf(datum, restExpr);
|
|
31
|
+
});
|
|
32
|
+
// If resulting structure is an array of array, flatten it
|
|
33
|
+
if (isArray(formattedData[0])) {
|
|
34
|
+
formattedData = flatten(formattedData);
|
|
35
|
+
}
|
|
36
|
+
listenerFn(formattedData, oldVal);
|
|
132
37
|
}
|
|
133
|
-
listenerFn(formattedData, oldVal);
|
|
134
38
|
};
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
const regex = /\[\$i\]/g;
|
|
39
|
+
const getUpdatedWatcInfo = (expr, acceptsArray, listener) => {
|
|
40
|
+
// listener doesn't accept array
|
|
41
|
+
// replace all `[$i]` with `[0]` and return the expression
|
|
42
|
+
let regex = /\[\$i\]/g, $I = '[$i]', $0 = '[0]';
|
|
140
43
|
if (!acceptsArray) {
|
|
141
44
|
return {
|
|
142
|
-
expr: expr.replace(regex,
|
|
143
|
-
listener
|
|
45
|
+
'expr': expr.replace(regex, $0),
|
|
46
|
+
'listener': listener
|
|
144
47
|
};
|
|
145
48
|
}
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
49
|
+
// listener accepts array
|
|
50
|
+
// replace all except the last `[$i]` with `[0]` and return the expression.
|
|
51
|
+
var index = expr.lastIndexOf($I), _expr = expr.substr(0, index).replace($I, $0), restExpr = expr.substr(index + 5), arrayConsumerFn = listener;
|
|
52
|
+
if (restExpr) {
|
|
53
|
+
arrayConsumerFn = arrayConsumer.bind(undefined, listener, restExpr);
|
|
54
|
+
}
|
|
152
55
|
return {
|
|
153
|
-
expr:
|
|
154
|
-
listener: arrayConsumerFn
|
|
56
|
+
'expr': _expr,
|
|
57
|
+
'listener': arrayConsumerFn
|
|
155
58
|
};
|
|
156
59
|
};
|
|
157
|
-
/**
|
|
158
|
-
* Determines if an expression is static (doesn't need to be watched)
|
|
159
|
-
*/
|
|
160
|
-
const STATIC_EXPRESSION_NAMES = [
|
|
161
|
-
"row.getProperty('investment')",
|
|
162
|
-
"row.getProperty('factsheetLink')",
|
|
163
|
-
"row.getProperty('isRebalanceEligible')"
|
|
164
|
-
];
|
|
165
|
-
const isStaticExpression = (expr) => {
|
|
166
|
-
if (typeof expr !== 'string') {
|
|
167
|
-
return false;
|
|
168
|
-
}
|
|
169
|
-
const trimmedExpr = expr.trim();
|
|
170
|
-
// Expressions that always evaluate to localization strings
|
|
171
|
-
// if (trimmedExpr.includes('appLocale')) {
|
|
172
|
-
// return true;
|
|
173
|
-
// }
|
|
174
|
-
// Hard-coded static expression names
|
|
175
|
-
if (STATIC_EXPRESSION_NAMES.includes(trimmedExpr)) {
|
|
176
|
-
return true;
|
|
177
|
-
}
|
|
178
|
-
return false;
|
|
179
|
-
};
|
|
180
|
-
/**
|
|
181
|
-
* Gets the scope type from the scope object
|
|
182
|
-
*/
|
|
183
|
-
const getScopeType = ($scope) => {
|
|
184
|
-
if (!$scope) {
|
|
185
|
-
return null;
|
|
186
|
-
}
|
|
187
|
-
if ($scope.pageName)
|
|
188
|
-
return 'Page';
|
|
189
|
-
if ($scope.prefabName)
|
|
190
|
-
return 'Prefab';
|
|
191
|
-
if ($scope.partialName)
|
|
192
|
-
return 'Partial';
|
|
193
|
-
// Check for App scope
|
|
194
|
-
if ($scope.Variables !== undefined &&
|
|
195
|
-
$scope.Actions !== undefined &&
|
|
196
|
-
!$scope.pageName &&
|
|
197
|
-
!$scope.prefabName &&
|
|
198
|
-
!$scope.partialName) {
|
|
199
|
-
return 'App';
|
|
200
|
-
}
|
|
201
|
-
if ($scope.constructor?.name === 'AppRef') {
|
|
202
|
-
return 'App';
|
|
203
|
-
}
|
|
204
|
-
return null;
|
|
205
|
-
};
|
|
206
|
-
/**
|
|
207
|
-
* Gets scope name based on scope type
|
|
208
|
-
*/
|
|
209
|
-
const getScopeName = ($scope, scopeType) => {
|
|
210
|
-
if (!scopeType || !$scope) {
|
|
211
|
-
return null;
|
|
212
|
-
}
|
|
213
|
-
switch (scopeType) {
|
|
214
|
-
case 'Prefab': return $scope.prefabName || null;
|
|
215
|
-
case 'Partial': return $scope.partialName || null;
|
|
216
|
-
case 'Page': return $scope.pageName || null;
|
|
217
|
-
default: return null;
|
|
218
|
-
}
|
|
219
|
-
};
|
|
220
|
-
/**
|
|
221
|
-
* Main watch function
|
|
222
|
-
*/
|
|
223
60
|
export const $watch = (expr, $scope, $locals, listener, identifier = watchIdGenerator.nextUid(), doNotClone = false, config = {}, isMuted) => {
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
const watchInfo = getUpdatedWatchInfo(expr, config.arrayType || config.isList || false, listener);
|
|
61
|
+
if (expr.indexOf('[$i]') !== -1) {
|
|
62
|
+
let watchInfo = getUpdatedWatcInfo(expr, config && (config.arrayType || config.isList), listener);
|
|
227
63
|
expr = watchInfo.expr;
|
|
228
64
|
listener = watchInfo.listener;
|
|
229
65
|
}
|
|
230
|
-
// Handle static expressions
|
|
231
|
-
if (isStaticExpression(expr)) {
|
|
232
|
-
try {
|
|
233
|
-
const fn = $parseExpr(expr);
|
|
234
|
-
const staticValue = fn($scope, $locals);
|
|
235
|
-
listener(staticValue, FIRST_TIME_WATCH);
|
|
236
|
-
}
|
|
237
|
-
catch (e) {
|
|
238
|
-
console.warn(`Error evaluating static expression '${expr}':`, e);
|
|
239
|
-
listener(undefined, FIRST_TIME_WATCH);
|
|
240
|
-
}
|
|
241
|
-
return () => { }; // No-op unsubscribe
|
|
242
|
-
}
|
|
243
66
|
const fn = $parseExpr(expr);
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
const destroyFn = () => $unwatch(identifier);
|
|
247
|
-
const watchInfo = {
|
|
248
|
-
fn: fn.bind(null, $scope, $locals),
|
|
67
|
+
registry.set(identifier, {
|
|
68
|
+
fn: fn.bind(expr, $scope, $locals),
|
|
249
69
|
listener,
|
|
250
70
|
expr,
|
|
251
71
|
last: FIRST_TIME_WATCH,
|
|
252
72
|
doNotClone,
|
|
253
|
-
isMuted
|
|
254
|
-
scopeType,
|
|
255
|
-
scopeName,
|
|
256
|
-
destroyFn
|
|
257
|
-
};
|
|
258
|
-
// Store in registry
|
|
259
|
-
const widgetId = getWidgetId(identifier);
|
|
260
|
-
if (widgetId) {
|
|
261
|
-
const propertyName = getPropertyName(identifier);
|
|
262
|
-
if (!registry.has(widgetId)) {
|
|
263
|
-
registry.set(widgetId, {});
|
|
264
|
-
}
|
|
265
|
-
const widgetGroup = registry.get(widgetId);
|
|
266
|
-
widgetGroup[propertyName] = watchInfo;
|
|
267
|
-
}
|
|
268
|
-
else {
|
|
269
|
-
registry.set(identifier, watchInfo);
|
|
270
|
-
}
|
|
271
|
-
return destroyFn;
|
|
272
|
-
};
|
|
273
|
-
/**
|
|
274
|
-
* Unwatches a single identifier
|
|
275
|
-
*/
|
|
276
|
-
export const $unwatch = (identifier) => {
|
|
277
|
-
const widgetId = getWidgetId(identifier);
|
|
278
|
-
if (widgetId) {
|
|
279
|
-
const propertyName = getPropertyName(identifier);
|
|
280
|
-
const widgetGroup = registry.get(widgetId);
|
|
281
|
-
//@ts-ignore
|
|
282
|
-
if (widgetGroup && typeof widgetGroup === 'object' && !widgetGroup.fn) {
|
|
283
|
-
const watchInfo = widgetGroup[propertyName];
|
|
284
|
-
if (watchInfo) {
|
|
285
|
-
delete widgetGroup[propertyName];
|
|
286
|
-
// Clean up empty widget groups
|
|
287
|
-
if (Object.keys(widgetGroup).length === 0) {
|
|
288
|
-
registry.delete(widgetId);
|
|
289
|
-
}
|
|
290
|
-
return true;
|
|
291
|
-
}
|
|
292
|
-
}
|
|
293
|
-
}
|
|
294
|
-
// Fallback to direct lookup
|
|
295
|
-
if (registry.has(identifier)) {
|
|
296
|
-
registry.delete(identifier);
|
|
297
|
-
return true;
|
|
298
|
-
}
|
|
299
|
-
return false;
|
|
300
|
-
};
|
|
301
|
-
/**
|
|
302
|
-
* Unwatches all watchers for a specific widget ID
|
|
303
|
-
*/
|
|
304
|
-
export const $unwatchAll = (widgetId) => {
|
|
305
|
-
if (!widgetId || typeof widgetId !== 'string') {
|
|
306
|
-
return 0;
|
|
307
|
-
}
|
|
308
|
-
const widgetGroup = registry.get(widgetId);
|
|
309
|
-
if (widgetGroup && typeof widgetGroup === 'object' && !widgetGroup.fn) {
|
|
310
|
-
const count = Object.keys(widgetGroup).length;
|
|
311
|
-
registry.delete(widgetId);
|
|
312
|
-
return count;
|
|
313
|
-
}
|
|
314
|
-
// Fallback: find all identifiers starting with this widget ID
|
|
315
|
-
let removedCount = 0;
|
|
316
|
-
const identifiersToRemove = [];
|
|
317
|
-
registry.forEach((_, key) => {
|
|
318
|
-
if (key.startsWith(widgetId + '_')) {
|
|
319
|
-
identifiersToRemove.push(key);
|
|
320
|
-
}
|
|
73
|
+
isMuted: isMuted
|
|
321
74
|
});
|
|
322
|
-
|
|
323
|
-
if ($unwatch(identifier)) {
|
|
324
|
-
removedCount++;
|
|
325
|
-
}
|
|
326
|
-
});
|
|
327
|
-
return removedCount;
|
|
75
|
+
return () => $unwatch(identifier);
|
|
328
76
|
};
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
if (!scopeType) {
|
|
335
|
-
return 0;
|
|
336
|
-
}
|
|
337
|
-
let removedCount = 0;
|
|
338
|
-
const identifiersToRemove = [];
|
|
339
|
-
registry.forEach((value, key) => {
|
|
340
|
-
// Handle grouped structure (widget groups)
|
|
341
|
-
if (value && typeof value === 'object' && !value.fn) {
|
|
342
|
-
Object.entries(value).forEach(([propertyName, watchInfo]) => {
|
|
343
|
-
if (watchInfo?.scopeType === scopeType &&
|
|
344
|
-
(!scopeName || watchInfo.scopeName === scopeName)) {
|
|
345
|
-
identifiersToRemove.push(`${key}_${propertyName}`);
|
|
346
|
-
}
|
|
347
|
-
});
|
|
348
|
-
}
|
|
349
|
-
else {
|
|
350
|
-
// Direct watchInfo
|
|
351
|
-
const watchInfo = value;
|
|
352
|
-
if (watchInfo?.scopeType === scopeType &&
|
|
353
|
-
(!scopeName || watchInfo.scopeName === scopeName)) {
|
|
354
|
-
identifiersToRemove.push(key);
|
|
355
|
-
}
|
|
356
|
-
}
|
|
357
|
-
});
|
|
358
|
-
// Unwatch all collected identifiers
|
|
359
|
-
identifiersToRemove.forEach(identifier => {
|
|
360
|
-
if ($unwatch(identifier)) {
|
|
361
|
-
removedCount++;
|
|
362
|
-
}
|
|
363
|
-
});
|
|
364
|
-
return removedCount;
|
|
365
|
-
};
|
|
366
|
-
/**
|
|
367
|
-
* Processes a single watch info during trigger cycle
|
|
368
|
-
*/
|
|
369
|
-
const processWatchInfo = (watchInfo) => {
|
|
370
|
-
if (!watchInfo?.fn || (watchInfo.isMuted?.() ?? false)) {
|
|
371
|
-
return false;
|
|
372
|
-
}
|
|
373
|
-
let newValue;
|
|
374
|
-
try {
|
|
375
|
-
newValue = watchInfo.fn();
|
|
376
|
-
}
|
|
377
|
-
catch (e) {
|
|
378
|
-
console.warn(`Error executing expression: '${watchInfo.expr}'`, e);
|
|
379
|
-
return false;
|
|
380
|
-
}
|
|
381
|
-
if (isEqual(newValue, watchInfo.last)) {
|
|
382
|
-
return false;
|
|
383
|
-
}
|
|
384
|
-
// Change detected
|
|
385
|
-
changedByWatch = true;
|
|
386
|
-
watchInfo.last = isObject(newValue) && !watchInfo.doNotClone
|
|
387
|
-
? clone(newValue)
|
|
388
|
-
: newValue;
|
|
389
|
-
watchInfo.listener(newValue, watchInfo.last === newValue ? FIRST_TIME_WATCH : watchInfo.last);
|
|
390
|
-
changedByWatch = false;
|
|
391
|
-
return true;
|
|
392
|
-
};
|
|
393
|
-
/**
|
|
394
|
-
* Triggers all watchers
|
|
395
|
-
*/
|
|
396
|
-
const triggerWatchers = (ignoreMuted = false) => {
|
|
77
|
+
export const $unwatch = identifier => registry.delete(identifier);
|
|
78
|
+
let changedByWatch = false;
|
|
79
|
+
const $RAF = window.requestAnimationFrame;
|
|
80
|
+
let ngZone;
|
|
81
|
+
const triggerWatchers = (ignoreMuted) => {
|
|
397
82
|
if (muted && !ignoreMuted) {
|
|
398
83
|
return;
|
|
399
84
|
}
|
|
85
|
+
const limit = 5;
|
|
400
86
|
let pass = 1;
|
|
401
87
|
let changeDetected;
|
|
402
88
|
do {
|
|
403
89
|
changeDetected = false;
|
|
404
|
-
registry.forEach(
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
Object.values(value).forEach((watchInfo) => {
|
|
408
|
-
if (processWatchInfo(watchInfo)) {
|
|
409
|
-
changeDetected = true;
|
|
410
|
-
}
|
|
411
|
-
});
|
|
90
|
+
registry.forEach(watchInfo => {
|
|
91
|
+
if (watchInfo.isMuted && watchInfo.isMuted()) {
|
|
92
|
+
return;
|
|
412
93
|
}
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
94
|
+
const fn = watchInfo.fn;
|
|
95
|
+
const listener = watchInfo.listener;
|
|
96
|
+
const ov = watchInfo.last;
|
|
97
|
+
let nv;
|
|
98
|
+
try {
|
|
99
|
+
nv = fn();
|
|
100
|
+
}
|
|
101
|
+
catch (e) {
|
|
102
|
+
console.warn(`error in executing expression: '${watchInfo.expr}'`);
|
|
103
|
+
}
|
|
104
|
+
if (!isEqual(nv, ov)) {
|
|
105
|
+
changeDetected = true;
|
|
106
|
+
changedByWatch = true;
|
|
107
|
+
watchInfo.last = nv;
|
|
108
|
+
// @ts-ignore
|
|
109
|
+
if (isObject(nv) && !watchInfo.doNotClone && nv.__cloneable__ !== false) {
|
|
110
|
+
watchInfo.last = clone(nv);
|
|
417
111
|
}
|
|
112
|
+
listener(nv, ov);
|
|
113
|
+
resetChangeFromWatch();
|
|
418
114
|
}
|
|
419
115
|
});
|
|
420
116
|
pass++;
|
|
421
|
-
} while (changeDetected && pass <
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
if (changeDetected && pass === MAX_WATCH_CYCLES) {
|
|
425
|
-
console.warn(`Watch cycles exceeded limit of ${MAX_WATCH_CYCLES}`);
|
|
117
|
+
} while (changeDetected && pass < limit);
|
|
118
|
+
if (changeDetected && pass === limit) {
|
|
119
|
+
console.warn(`Number of watch cycles gone above set limit of: ${limit} `);
|
|
426
120
|
}
|
|
427
121
|
};
|
|
428
|
-
|
|
429
|
-
export const
|
|
430
|
-
ngZone = zone;
|
|
431
|
-
};
|
|
432
|
-
export const setAppRef = (ref) => {
|
|
122
|
+
export const setNgZone = zone => ngZone = zone;
|
|
123
|
+
export const setAppRef = ref => {
|
|
433
124
|
appRef = ref;
|
|
434
125
|
};
|
|
435
126
|
export const isChangeFromWatch = () => changedByWatch;
|
|
436
|
-
export const resetChangeFromWatch = () =>
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
// Debounced trigger
|
|
127
|
+
export const resetChangeFromWatch = () => changedByWatch = false;
|
|
128
|
+
window.watchRegistry = registry;
|
|
129
|
+
let skipWatchers;
|
|
440
130
|
const debouncedTriggerWatchers = debounce(() => {
|
|
441
131
|
skipWatchers = true;
|
|
442
132
|
ngZone.run(() => triggerWatchers());
|
|
443
|
-
},
|
|
444
|
-
export const $invokeWatchers = (force
|
|
133
|
+
}, 100);
|
|
134
|
+
export const $invokeWatchers = (force, ignoreMuted) => {
|
|
445
135
|
if (force) {
|
|
446
136
|
triggerWatchers(ignoreMuted);
|
|
447
137
|
}
|
|
@@ -455,8 +145,7 @@ export const $invokeWatchers = (force = false, ignoreMuted = false) => {
|
|
|
455
145
|
};
|
|
456
146
|
export const $appDigest = (() => {
|
|
457
147
|
let queued = false;
|
|
458
|
-
|
|
459
|
-
return (force = false) => {
|
|
148
|
+
return (force) => {
|
|
460
149
|
if (!appRef) {
|
|
461
150
|
return;
|
|
462
151
|
}
|
|
@@ -468,14 +157,14 @@ export const $appDigest = (() => {
|
|
|
468
157
|
if (queued) {
|
|
469
158
|
return;
|
|
470
159
|
}
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
160
|
+
else {
|
|
161
|
+
queued = true;
|
|
162
|
+
$RAF(() => {
|
|
163
|
+
ngZone.run(() => appRef.tick());
|
|
164
|
+
queued = false;
|
|
165
|
+
});
|
|
166
|
+
}
|
|
476
167
|
}
|
|
477
168
|
};
|
|
478
169
|
})();
|
|
479
|
-
// Export registry for debugging
|
|
480
|
-
window.watchRegistry = registry;
|
|
481
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"watcher.js","sourceRoot":"","sources":["../../../../projects/core/src/utils/watcher.ts"],"names":[],"mappings":"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","sourcesContent":["import { IDGenerator } from './id-generator';\nimport { $parseExpr } from './expression-parser';\nimport { findValueOf } from './utils';\nimport { clone, flatten, isArray, isEqual, isObject } from \"lodash-es\";\n\n// Constants\nconst WIDGET_ID_REGEX = /^(widget-[^_]+)/;\nconst WIDGET_PROPERTY_REGEX = /^widget-[^_]+_(.+)$/;\nconst ARRAY_INDEX_PLACEHOLDER = '[$i]';\nconst ARRAY_INDEX_ZERO = '[0]';\nconst DEBOUNCE_WAIT = 100;\nconst MAX_WATCH_CYCLES = 5;\n\nconst registry = new Map<string, any>();\nconst watchIdGenerator = new IDGenerator('watch-id-');\n\nexport const FIRST_TIME_WATCH = Object.freeze({});\n\n// Type definitions\ninterface WatchInfo {\n    fn: () => any;\n    listener: (newVal: any, oldVal: any) => void;\n    expr: string;\n    last: any;\n    doNotClone: boolean;\n    isMuted?: () => boolean;\n    scopeType: ScopeType | null;\n    scopeName: string | null;\n    destroyFn?: () => void;\n}\n\ntype ScopeType = 'Page' | 'Prefab' | 'Partial' | 'App';\n\ninterface WatchConfig {\n    arrayType?: boolean;\n    isList?: boolean;\n}\n\n// State\nlet muted = false;\nlet changedByWatch = false;\nlet skipWatchers = false;\nlet ngZone: any;\nlet appRef: any;\n\n/********************************************************************\n * CLEANUP SCHEDULER WITH:\n *  - Significant watcher delta trigger (+/-)\n *  - UI stabilization delay (1–2 seconds)\n *  - Cooldown to prevent repeated scheduling\n ********************************************************************/\nconst CLEANUP_TRIGGER_DELTA = 300;   // watcher change threshold\nconst CLEANUP_DELAY = 1500;          // delay before running cleanup\nconst CLEANUP_COOLDOWN = 4000;       // prevent re-scheduling for 4s\n\nlet lastWatcherCount = 0;\nlet scheduledCleanup: any = null;\nlet lastScheduledTime = 0;\n\nfunction getWatcherCount(): number {\n    let count = 0;\n    registry.forEach(bucket => count += bucket.size);\n    return count;\n}\n\nexport const cleanupStaleWatchers = (): number => {\n    // console.log(\".........Cleaning up stale watchers...registry.size....\", registry.size);\n    let removed = 0;\n\n    registry.forEach((bucket, widgetId) => {\n        if (!document.querySelector(`[widget-id=\"${widgetId}\"]`)) {\n            for (const key in bucket) {\n                if (bucket.hasOwnProperty(key) && key !== \"scopeType\" && key !== \"scopeName\" && typeof bucket[key] !== \"function\") {\n                    let watchInfo = bucket[key];\n                    if (watchInfo && watchInfo.destroyFn) {\n                        watchInfo.destroyFn();\n                    }\n                }\n            }\n            removed++;\n            registry.delete(widgetId);\n            return;\n        }\n    });\n    return removed;\n};\n\nexport const scheduleThresholdCleanup = (): void => {\n    const now = performance.now();\n    let lastTriggerPeriod = now - lastScheduledTime;\n    // If a cleanup was scheduled recently, skip scheduling\n    if (lastTriggerPeriod < CLEANUP_COOLDOWN) {\n        return;\n    }\n\n    const current = getWatcherCount();\n    const delta = Math.abs(current - lastWatcherCount);  // significant + or -\n\n    // If change not large enough, skip scheduling\n    if (delta <= CLEANUP_TRIGGER_DELTA) {\n        lastWatcherCount = current;\n        return;\n    }\n\n    // Prevent re-scheduling: set timestamp early\n    lastScheduledTime = now;\n\n    // Clear previous scheduled cleanup (if any)\n    if (scheduledCleanup) {\n        clearTimeout(scheduledCleanup);\n    }\n\n    // Schedule cleanup after UI stabilizes (delay prevents aggressive cleanup)\n    scheduledCleanup = setTimeout(() => {\n        cleanupStaleWatchers();\n        scheduledCleanup = null;\n    }, CLEANUP_DELAY);\n\n    lastWatcherCount = current;\n};\n\n// Utility functions\nexport const isFirstTimeChange = (v: any): boolean => v === FIRST_TIME_WATCH;\n\nexport const debounce = (fn: Function, wait: number = DEBOUNCE_WAIT) => {\n    let timeout: any;\n    return (...args: any[]) => {\n        window['__zone_symbol__clearTimeout'](timeout);\n        timeout = window['__zone_symbol__setTimeout'](() => fn(...args), wait);\n    };\n};\n\nexport const muteWatchers = (): void => {\n    muted = true;\n};\n\nexport const unMuteWatchers = (): void => {\n    muted = false;\n    triggerWatchers();\n};\n\n/**\n * Extracts widget ID from identifier (e.g., \"widget-id23_eventsource\" -> \"widget-id23\")\n */\nconst getWidgetId = (identifier: string): string | null => {\n    if (!identifier || typeof identifier !== 'string') {\n        return null;\n    }\n    const match = identifier.match(WIDGET_ID_REGEX);\n    return match ? match[1] : null;\n};\n\n/**\n * Extracts property name from identifier (e.g., \"widget-id23_eventsource\" -> \"eventsource\")\n */\nconst getPropertyName = (identifier: string): string => {\n    if (!identifier || typeof identifier !== 'string') {\n        return identifier;\n    }\n    const match = identifier.match(WIDGET_PROPERTY_REGEX);\n    return match ? match[1] : identifier;\n};\n\n/**\n * Array consumer wrapper for array-based expressions\n */\nconst arrayConsumer = (listenerFn: any, restExpr: string, newVal: any, oldVal: any): void => {\n    if (!isArray(newVal)) {\n        return;\n    }\n\n    let formattedData = newVal.map(datum => findValueOf(datum, restExpr));\n\n    // Flatten if result is array of arrays\n    if (isArray(formattedData[0])) {\n        formattedData = flatten(formattedData);\n    }\n\n    listenerFn(formattedData, oldVal);\n};\n\n/**\n * Updates watch info for array expressions\n */\nconst getUpdatedWatchInfo = (expr: string, acceptsArray: boolean, listener: any) => {\n    const regex = /\\[\\$i\\]/g;\n\n    if (!acceptsArray) {\n        return {\n            expr: expr.replace(regex, ARRAY_INDEX_ZERO),\n            listener\n        };\n    }\n\n    const lastIndex = expr.lastIndexOf(ARRAY_INDEX_PLACEHOLDER);\n    const baseExpr = expr.substring(0, lastIndex).replace(ARRAY_INDEX_PLACEHOLDER, ARRAY_INDEX_ZERO);\n    const restExpr = expr.substring(lastIndex + 5);\n\n    const arrayConsumerFn = restExpr\n        ? arrayConsumer.bind(undefined, listener, restExpr)\n        : listener;\n\n    return {\n        expr: baseExpr,\n        listener: arrayConsumerFn\n    };\n};\n\n/**\n * Determines if an expression is static (doesn't need to be watched)\n */\nconst STATIC_EXPRESSION_NAMES = [\n    \"row.getProperty('investment')\",\n    \"row.getProperty('factsheetLink')\",\n    \"row.getProperty('isRebalanceEligible')\"\n];\n\nconst isStaticExpression = (expr: any): boolean => {\n    if (typeof expr !== 'string') {\n        return false;\n    }\n\n    const trimmedExpr = expr.trim();\n\n    // Expressions that always evaluate to localization strings\n    // if (trimmedExpr.includes('appLocale')) {\n    //     return true;\n    // }\n\n    // Hard-coded static expression names\n    if (STATIC_EXPRESSION_NAMES.includes(trimmedExpr)) {\n        return true;\n    }\n\n    return false;\n};\n\n/**\n * Gets the scope type from the scope object\n */\nconst getScopeType = ($scope: any): ScopeType | null => {\n    if (!$scope) {\n        return null;\n    }\n\n    if ($scope.pageName) return 'Page';\n    if ($scope.prefabName) return 'Prefab';\n    if ($scope.partialName) return 'Partial';\n\n    // Check for App scope\n    if ($scope.Variables !== undefined &&\n        $scope.Actions !== undefined &&\n        !$scope.pageName &&\n        !$scope.prefabName &&\n        !$scope.partialName) {\n        return 'App';\n    }\n\n    if ($scope.constructor?.name === 'AppRef') {\n        return 'App';\n    }\n\n    return null;\n};\n\n/**\n * Gets scope name based on scope type\n */\nconst getScopeName = ($scope: any, scopeType: ScopeType | null): string | null => {\n    if (!scopeType || !$scope) {\n        return null;\n    }\n\n    switch (scopeType) {\n        case 'Prefab': return $scope.prefabName || null;\n        case 'Partial': return $scope.partialName || null;\n        case 'Page': return $scope.pageName || null;\n        default: return null;\n    }\n};\n\n/**\n * Main watch function\n */\nexport const $watch = (\n    expr: string,\n    $scope: any,\n    $locals: any,\n    listener: any,\n    identifier: string = watchIdGenerator.nextUid(),\n    doNotClone: boolean = false,\n    config: WatchConfig = {},\n    isMuted?: () => boolean\n) => {\n    // Handle array expressions\n    if (expr.includes(ARRAY_INDEX_PLACEHOLDER)) {\n        const watchInfo = getUpdatedWatchInfo(expr, config.arrayType || config.isList || false, listener);\n        expr = watchInfo.expr;\n        listener = watchInfo.listener;\n    }\n\n    // Handle static expressions\n    if (isStaticExpression(expr)) {\n        try {\n            const fn = $parseExpr(expr);\n            const staticValue = fn($scope, $locals);\n            listener(staticValue, FIRST_TIME_WATCH);\n        } catch (e) {\n            console.warn(`Error evaluating static expression '${expr}':`, e);\n            listener(undefined, FIRST_TIME_WATCH);\n        }\n        return () => { }; // No-op unsubscribe\n    }\n\n    const fn = $parseExpr(expr);\n    const scopeType = getScopeType($scope);\n    const scopeName = getScopeName($scope, scopeType);\n\n    const destroyFn = () => $unwatch(identifier);\n    const watchInfo: WatchInfo = {\n        fn: fn.bind(null, $scope, $locals),\n        listener,\n        expr,\n        last: FIRST_TIME_WATCH,\n        doNotClone,\n        isMuted,\n        scopeType,\n        scopeName,\n        destroyFn\n    };\n\n    // Store in registry\n    const widgetId = getWidgetId(identifier);\n\n    if (widgetId) {\n        const propertyName = getPropertyName(identifier);\n        if (!registry.has(widgetId)) {\n            registry.set(widgetId, {});\n        }\n        const widgetGroup = registry.get(widgetId);\n        widgetGroup[propertyName] = watchInfo;\n    } else {\n        registry.set(identifier, watchInfo);\n    }\n\n    return destroyFn;\n};\n\n/**\n * Unwatches a single identifier\n */\nexport const $unwatch = (identifier: string): boolean => {\n    const widgetId = getWidgetId(identifier);\n\n    if (widgetId) {\n        const propertyName = getPropertyName(identifier);\n        const widgetGroup = registry.get(widgetId);\n\n        //@ts-ignore\n        if (widgetGroup && typeof widgetGroup === 'object' && !widgetGroup.fn) {\n            const watchInfo = widgetGroup[propertyName];\n            if (watchInfo) {\n                delete widgetGroup[propertyName];\n                // Clean up empty widget groups\n                if (Object.keys(widgetGroup).length === 0) {\n                    registry.delete(widgetId);\n                }\n                return true;\n            }\n        }\n    }\n\n    // Fallback to direct lookup\n    if (registry.has(identifier)) {\n        registry.delete(identifier);\n        return true;\n    }\n\n    return false;\n};\n\n/**\n * Unwatches all watchers for a specific widget ID\n */\nexport const $unwatchAll = (widgetId: string): number => {\n    if (!widgetId || typeof widgetId !== 'string') {\n        return 0;\n    }\n\n    const widgetGroup = registry.get(widgetId);\n\n    if (widgetGroup && typeof widgetGroup === 'object' && !widgetGroup.fn) {\n        const count = Object.keys(widgetGroup).length;\n        registry.delete(widgetId);\n        return count;\n    }\n\n    // Fallback: find all identifiers starting with this widget ID\n    let removedCount = 0;\n    const identifiersToRemove: string[] = [];\n\n    registry.forEach((_, key) => {\n        if (key.startsWith(widgetId + '_')) {\n            identifiersToRemove.push(key);\n        }\n    });\n\n    identifiersToRemove.forEach(identifier => {\n        if ($unwatch(identifier)) {\n            removedCount++;\n        }\n    });\n\n    return removedCount;\n};\n\n/**\n * Unwatches all watchers for a specific scope (Page, Prefab, Partial, or App)\n * Now works directly with the main registry instead of separate scoped registries\n */\nexport const $unwatchAllByScope = (scopeType: ScopeType, scopeName?: string): number => {\n    if (!scopeType) {\n        return 0;\n    }\n\n    let removedCount = 0;\n    const identifiersToRemove: string[] = [];\n\n    registry.forEach((value, key) => {\n        // Handle grouped structure (widget groups)\n        if (value && typeof value === 'object' && !value.fn) {\n            Object.entries(value).forEach(([propertyName, watchInfo]: [string, any]) => {\n                if (watchInfo?.scopeType === scopeType &&\n                    (!scopeName || watchInfo.scopeName === scopeName)) {\n                    identifiersToRemove.push(`${key}_${propertyName}`);\n                }\n            });\n        } else {\n            // Direct watchInfo\n            const watchInfo = value as WatchInfo;\n            if (watchInfo?.scopeType === scopeType &&\n                (!scopeName || watchInfo.scopeName === scopeName)) {\n                identifiersToRemove.push(key);\n            }\n        }\n    });\n\n    // Unwatch all collected identifiers\n    identifiersToRemove.forEach(identifier => {\n        if ($unwatch(identifier)) {\n            removedCount++;\n        }\n    });\n\n    return removedCount;\n};\n\n/**\n * Processes a single watch info during trigger cycle\n */\nconst processWatchInfo = (watchInfo: WatchInfo): boolean => {\n    if (!watchInfo?.fn || (watchInfo.isMuted?.() ?? false)) {\n        return false;\n    }\n\n    let newValue: any;\n    try {\n        newValue = watchInfo.fn();\n    } catch (e) {\n        console.warn(`Error executing expression: '${watchInfo.expr}'`, e);\n        return false;\n    }\n\n    if (isEqual(newValue, watchInfo.last)) {\n        return false;\n    }\n\n    // Change detected\n    changedByWatch = true;\n    watchInfo.last = isObject(newValue) && !watchInfo.doNotClone\n        ? clone(newValue)\n        : newValue;\n\n    watchInfo.listener(newValue, watchInfo.last === newValue ? FIRST_TIME_WATCH : watchInfo.last);\n    changedByWatch = false;\n\n    return true;\n};\n\n/**\n * Triggers all watchers\n */\nconst triggerWatchers = (ignoreMuted: boolean = false): void => {\n    if (muted && !ignoreMuted) {\n        return;\n    }\n\n    let pass = 1;\n    let changeDetected: boolean;\n\n    do {\n        changeDetected = false;\n\n        registry.forEach((value) => {\n            // Handle grouped structure\n            if (value && typeof value === 'object' && !value.fn) {\n                Object.values(value).forEach((watchInfo: any) => {\n                    if (processWatchInfo(watchInfo)) {\n                        changeDetected = true;\n                    }\n                });\n            } else {\n                // Direct watchInfo\n                if (processWatchInfo(value as WatchInfo)) {\n                    changeDetected = true;\n                }\n            }\n        });\n\n        pass++;\n    } while (changeDetected && pass < MAX_WATCH_CYCLES);\n\n    // Schedule cleanup after watchers are triggered\n    scheduleThresholdCleanup();\n\n    if (changeDetected && pass === MAX_WATCH_CYCLES) {\n        console.warn(`Watch cycles exceeded limit of ${MAX_WATCH_CYCLES}`);\n    }\n};\n\n// Angular zone integration\nexport const setNgZone = (zone: any): void => {\n    ngZone = zone;\n};\n\nexport const setAppRef = (ref: any): void => {\n    appRef = ref;\n};\n\nexport const isChangeFromWatch = (): boolean => changedByWatch;\nexport const resetChangeFromWatch = (): void => {\n    changedByWatch = false;\n};\n\n// Debounced trigger\nconst debouncedTriggerWatchers = debounce(() => {\n    skipWatchers = true;\n    ngZone.run(() => triggerWatchers());\n}, DEBOUNCE_WAIT);\n\nexport const $invokeWatchers = (force: boolean = false, ignoreMuted: boolean = false): void => {\n    if (force) {\n        triggerWatchers(ignoreMuted);\n    } else {\n        if (skipWatchers) {\n            skipWatchers = false;\n            return;\n        }\n        debouncedTriggerWatchers();\n    }\n};\n\nexport const $appDigest = (() => {\n    let queued = false;\n    const $RAF = window.requestAnimationFrame;\n\n    return (force: boolean = false): void => {\n        if (!appRef) {\n            return;\n        }\n\n        if (force) {\n            ngZone.run(() => appRef.tick());\n            queued = false;\n        } else {\n            if (queued) {\n                return;\n            }\n            queued = true;\n            $RAF(() => {\n                ngZone.run(() => appRef.tick());\n                queued = false;\n            });\n        }\n    };\n})();\n\n// Export registry for debugging\n(window as any).watchRegistry = registry;\n"]}
|
|
170
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"watcher.js","sourceRoot":"","sources":["../../../../projects/core/src/utils/watcher.ts"],"names":[],"mappings":"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","sourcesContent":["import { IDGenerator } from './id-generator';\n\nimport { $parseExpr } from './expression-parser';\nimport { findValueOf } from './utils';\nimport {clone, flatten, isArray, isEqual, isObject} from \"lodash-es\";\n\nconst registry = new Map<string, any>();\n\nconst watchIdGenerator = new IDGenerator('watch-id-');\n\nexport const FIRST_TIME_WATCH = {};\n\nObject.freeze(FIRST_TIME_WATCH);\n\nexport const isFirstTimeChange = v => v === FIRST_TIME_WATCH;\n\nlet muted = false;\nlet appRef;\n\nexport const debounce = (fn: Function, wait: number = 50) => {\n    let timeout;\n    return (...args) => {\n        window['__zone_symbol__clearTimeout'](timeout);\n        timeout = window['__zone_symbol__setTimeout'](() => fn(...args), wait);\n    };\n};\n\nexport const muteWatchers = () => {\n    muted = true;\n};\n\nexport const unMuteWatchers = () => {\n    muted = false;\n    triggerWatchers();\n};\n\nconst arrayConsumer = (listenerFn, restExpr, newVal, oldVal) => {\n    let data = newVal,\n        formattedData;\n\n    if (isArray(data)) {\n        formattedData = data.map(function (datum) {\n            return findValueOf(datum, restExpr);\n        });\n\n        // If resulting structure is an array of array, flatten it\n        if (isArray(formattedData[0])) {\n            formattedData = flatten(formattedData);\n        }\n\n        listenerFn(formattedData, oldVal);\n    }\n};\n\nconst getUpdatedWatcInfo = (expr, acceptsArray, listener) => {\n    // listener doesn't accept array\n    // replace all `[$i]` with `[0]` and return the expression\n    let regex = /\\[\\$i\\]/g,\n        $I                = '[$i]',\n        $0                = '[0]';\n    if (!acceptsArray) {\n        return {\n            'expr'     : expr.replace(regex, $0),\n            'listener' : listener\n        };\n    }\n\n    // listener accepts array\n    // replace all except the last `[$i]` with `[0]` and return the expression.\n    var index           = expr.lastIndexOf($I),\n        _expr           = expr.substr(0, index).replace($I, $0),\n        restExpr        = expr.substr(index + 5),\n        arrayConsumerFn = listener;\n\n    if (restExpr) {\n        arrayConsumerFn = arrayConsumer.bind(undefined, listener, restExpr);\n    }\n\n    return {\n        'expr'     : _expr,\n        'listener' : arrayConsumerFn\n    };\n};\n\nexport const $watch = (expr, $scope, $locals, listener, identifier = watchIdGenerator.nextUid(), doNotClone = false, config:any={}, isMuted?: () => boolean) => {\n    if (expr.indexOf('[$i]') !== -1) {\n        let watchInfo = getUpdatedWatcInfo(expr, config && (config.arrayType || config.isList), listener);\n        expr = watchInfo.expr;\n        listener = watchInfo.listener;\n    }\n    const fn = $parseExpr(expr);\n\n    registry.set(identifier, {\n        fn: fn.bind(expr, $scope, $locals),\n        listener,\n        expr,\n        last: FIRST_TIME_WATCH,\n        doNotClone,\n        isMuted: isMuted\n    });\n\n    return () => $unwatch(identifier);\n};\n\nexport const $unwatch = identifier => registry.delete(identifier);\n\nlet changedByWatch = false;\n\nconst $RAF = window.requestAnimationFrame;\n\nlet ngZone;\n\nconst triggerWatchers = (ignoreMuted?: boolean) => {\n\n    if (muted && !ignoreMuted) {\n        return;\n    }\n\n    const limit = 5;\n    let pass = 1;\n    let changeDetected;\n\n    do {\n        changeDetected = false;\n        registry.forEach(watchInfo => {\n            if(watchInfo.isMuted && watchInfo.isMuted()) {\n                return;\n            }\n            const fn = watchInfo.fn;\n            const listener = watchInfo.listener;\n            const ov = watchInfo.last;\n            let nv;\n\n            try {\n                nv = fn();\n            } catch (e) {\n                console.warn(`error in executing expression: '${watchInfo.expr}'`);\n            }\n\n            if (!isEqual(nv, ov)) {\n                changeDetected = true;\n                changedByWatch = true;\n                watchInfo.last = nv;\n\n                // @ts-ignore\n                if (isObject(nv) && !watchInfo.doNotClone && nv.__cloneable__ !== false) {\n                    watchInfo.last = clone(nv);\n                }\n                listener(nv, ov);\n                resetChangeFromWatch();\n            }\n        });\n        pass++;\n\n    } while (changeDetected && pass < limit);\n\n    if (changeDetected && pass === limit) {\n        console.warn(`Number of watch cycles gone above set limit of: ${limit} `);\n    }\n};\n\nexport const setNgZone = zone => ngZone = zone;\n\nexport const setAppRef = ref => {\n    appRef = ref\n};\n\nexport const isChangeFromWatch = () => changedByWatch;\nexport const resetChangeFromWatch = () => changedByWatch = false;\n\n(<any>window).watchRegistry = registry;\n\nlet skipWatchers;\n\nconst debouncedTriggerWatchers = debounce(() => {\n    skipWatchers = true;\n    ngZone.run(() => triggerWatchers());\n}, 100);\n\nexport const $invokeWatchers = (force?: boolean, ignoreMuted?: boolean) => {\n    if (force) {\n        triggerWatchers(ignoreMuted);\n    } else {\n\n        if (skipWatchers) {\n            skipWatchers = false;\n            return;\n        }\n        debouncedTriggerWatchers();\n    }\n};\n\nexport const $appDigest = (() => {\n    let queued = false;\n    return (force?: boolean) => {\n        if (!appRef) {\n            return;\n        }\n        if (force) {\n            ngZone.run(() => appRef.tick());\n            queued = false;\n        } else {\n            if (queued) {\n                return;\n            } else {\n                queued = true;\n                $RAF(() => {\n                    ngZone.run(() => appRef.tick());\n                    queued = false;\n                });\n            }\n        }\n    };\n})();\n"]}
|