angular-three 1.9.16 → 1.10.1
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/esm2022/lib/canvas.mjs +243 -0
- package/esm2022/lib/directives/args.mjs +36 -0
- package/esm2022/lib/directives/common.mjs +32 -0
- package/esm2022/lib/directives/parent.mjs +36 -0
- package/esm2022/lib/directives/repeat.mjs +18 -0
- package/esm2022/lib/loader.mjs +55 -0
- package/esm2022/lib/pipes/push.mjs +50 -0
- package/esm2022/lib/portal.mjs +240 -0
- package/esm2022/lib/renderer/renderer.mjs +361 -0
- package/esm2022/lib/routed-scene.mjs +29 -0
- package/esm2022/lib/stores/rx-store.mjs +108 -0
- package/esm2022/lib/stores/store.mjs +404 -0
- package/esm2022/lib/types.mjs +2 -0
- package/esm2022/lib/utils/apply-props.mjs +107 -0
- package/esm2022/lib/utils/is.mjs +51 -0
- package/{fesm2020 → fesm2022}/angular-three.mjs +78 -61
- package/fesm2022/angular-three.mjs.map +1 -0
- package/lib/canvas.d.ts +1 -1
- package/lib/directives/args.d.ts +1 -1
- package/lib/directives/parent.d.ts +1 -1
- package/lib/directives/repeat.d.ts +1 -1
- package/lib/loader.d.ts +3 -11
- package/lib/portal.d.ts +2 -2
- package/lib/types.d.ts +91 -120
- package/lib/utils/is.d.ts +6 -1
- package/metadata.json +1 -0
- package/package.json +13 -17
- package/plugin/package.json +1 -4
- package/plugin/src/generators/init/compat.js +1 -1
- package/plugin/src/generators/init/compat.js.map +1 -1
- package/plugin/src/generators/init/init.d.ts +3 -3
- package/plugin/src/generators/init/init.js +23 -3
- package/plugin/src/generators/init/init.js.map +1 -1
- package/web-types.json +1 -0
- package/esm2020/lib/canvas.mjs +0 -242
- package/esm2020/lib/directives/args.mjs +0 -35
- package/esm2020/lib/directives/common.mjs +0 -31
- package/esm2020/lib/directives/parent.mjs +0 -35
- package/esm2020/lib/directives/repeat.mjs +0 -17
- package/esm2020/lib/loader.mjs +0 -56
- package/esm2020/lib/pipes/push.mjs +0 -49
- package/esm2020/lib/portal.mjs +0 -237
- package/esm2020/lib/renderer/renderer.mjs +0 -360
- package/esm2020/lib/routed-scene.mjs +0 -28
- package/esm2020/lib/stores/rx-store.mjs +0 -107
- package/esm2020/lib/stores/store.mjs +0 -402
- package/esm2020/lib/types.mjs +0 -2
- package/esm2020/lib/utils/apply-props.mjs +0 -91
- package/esm2020/lib/utils/is.mjs +0 -50
- package/fesm2015/angular-three.mjs +0 -3051
- package/fesm2015/angular-three.mjs.map +0 -1
- package/fesm2020/angular-three.mjs.map +0 -1
- /package/{esm2020 → esm2022}/angular-three.mjs +0 -0
- /package/{esm2020 → esm2022}/index.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/di/before-render.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/di/catalogue.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/di/destroy.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/di/ref.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/di/run-in-context.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/events.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/loop.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/renderer/di.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/renderer/enums.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/renderer/provider.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/renderer/store.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/renderer/utils.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/utils/attach.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/utils/instance.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/utils/make.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/utils/safe-detect-changes.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/utils/update.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/web/events.mjs +0 -0
|
@@ -0,0 +1,361 @@
|
|
|
1
|
+
import { DOCUMENT } from '@angular/common';
|
|
2
|
+
import { ChangeDetectorRef, getDebugNode, inject, Injectable, RendererFactory2, } from '@angular/core';
|
|
3
|
+
import { take } from 'rxjs';
|
|
4
|
+
import { NGT_CATALOGUE } from '../di/catalogue';
|
|
5
|
+
import { NgtStore } from '../stores/store';
|
|
6
|
+
import { getLocalState, prepare } from '../utils/instance';
|
|
7
|
+
import { is } from '../utils/is';
|
|
8
|
+
import { safeDetectChanges } from '../utils/safe-detect-changes';
|
|
9
|
+
import { NGT_COMPOUND_PREFIXES } from './di';
|
|
10
|
+
import { NgtRendererStore } from './store';
|
|
11
|
+
import { attachThreeChild, kebabToPascal, processThreeEvent, removeThreeChild, SPECIAL_DOM_TAG } from './utils';
|
|
12
|
+
import * as i0 from "@angular/core";
|
|
13
|
+
class NgtRendererFactory {
|
|
14
|
+
constructor() {
|
|
15
|
+
this.delegateRendererFactory = inject(RendererFactory2, { skipSelf: true });
|
|
16
|
+
this.catalogue = inject(NGT_CATALOGUE);
|
|
17
|
+
this.rendererMap = new Map();
|
|
18
|
+
this.routedSet = new Set();
|
|
19
|
+
// all Renderer instances share the same Store
|
|
20
|
+
this.rendererStore = new NgtRendererStore({
|
|
21
|
+
store: inject(NgtStore),
|
|
22
|
+
cdr: inject(ChangeDetectorRef),
|
|
23
|
+
portals: [],
|
|
24
|
+
compoundPrefixes: inject(NGT_COMPOUND_PREFIXES),
|
|
25
|
+
document: inject(DOCUMENT),
|
|
26
|
+
});
|
|
27
|
+
}
|
|
28
|
+
createRenderer(hostElement, type) {
|
|
29
|
+
const delegateRenderer = this.delegateRendererFactory.createRenderer(hostElement, type);
|
|
30
|
+
if (!type)
|
|
31
|
+
return delegateRenderer;
|
|
32
|
+
if (type['type']['isRoutedScene']) {
|
|
33
|
+
this.routedSet.add(type.id);
|
|
34
|
+
}
|
|
35
|
+
let renderer = this.rendererMap.get(type.id);
|
|
36
|
+
if (!renderer) {
|
|
37
|
+
renderer = new NgtRenderer(delegateRenderer, this.rendererStore, this.catalogue,
|
|
38
|
+
// setting root scene if there's no routed scene OR this component is the routed Scene
|
|
39
|
+
!hostElement && (this.routedSet.size === 0 || this.routedSet.has(type.id)));
|
|
40
|
+
this.rendererMap.set(type.id, renderer);
|
|
41
|
+
}
|
|
42
|
+
return renderer;
|
|
43
|
+
}
|
|
44
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtRendererFactory, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
45
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtRendererFactory }); }
|
|
46
|
+
}
|
|
47
|
+
export { NgtRendererFactory };
|
|
48
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtRendererFactory, decorators: [{
|
|
49
|
+
type: Injectable
|
|
50
|
+
}] });
|
|
51
|
+
/**
|
|
52
|
+
* Anything abbreviated with rS/RS stands for RendererState
|
|
53
|
+
*/
|
|
54
|
+
export class NgtRenderer {
|
|
55
|
+
constructor(delegate, store, catalogue, root = true) {
|
|
56
|
+
this.delegate = delegate;
|
|
57
|
+
this.store = store;
|
|
58
|
+
this.catalogue = catalogue;
|
|
59
|
+
this.root = root;
|
|
60
|
+
this.createText = this.delegate.createText.bind(this.delegate);
|
|
61
|
+
this.destroy = this.delegate.destroy.bind(this.delegate);
|
|
62
|
+
this.destroyNode = null;
|
|
63
|
+
this.selectRootElement = this.delegate.selectRootElement.bind(this.delegate);
|
|
64
|
+
this.nextSibling = this.delegate.nextSibling.bind(this.delegate);
|
|
65
|
+
this.removeAttribute = this.delegate.removeAttribute.bind(this.delegate);
|
|
66
|
+
this.addClass = this.delegate.addClass.bind(this.delegate);
|
|
67
|
+
this.removeClass = this.delegate.removeClass.bind(this.delegate);
|
|
68
|
+
this.setStyle = this.delegate.setStyle.bind(this.delegate);
|
|
69
|
+
this.removeStyle = this.delegate.removeStyle.bind(this.delegate);
|
|
70
|
+
this.setValue = this.delegate.setValue.bind(this.delegate);
|
|
71
|
+
}
|
|
72
|
+
createElement(name, namespace) {
|
|
73
|
+
const element = this.delegate.createElement(name, namespace);
|
|
74
|
+
// on first pass, we return the Root Scene as the root node
|
|
75
|
+
if (this.root) {
|
|
76
|
+
this.root = false;
|
|
77
|
+
const node = this.store.createNode('three', this.store.rootScene);
|
|
78
|
+
node.__ngt_renderer__[14 /* NgtRendererClassId.injectorFactory */] = () => getDebugNode(element).injector;
|
|
79
|
+
return node;
|
|
80
|
+
}
|
|
81
|
+
// handle compound
|
|
82
|
+
if (this.store.isCompound(name)) {
|
|
83
|
+
return this.store.createNode('compound', element);
|
|
84
|
+
}
|
|
85
|
+
// handle portal
|
|
86
|
+
if (name === SPECIAL_DOM_TAG.NGT_PORTAL) {
|
|
87
|
+
return this.store.createNode('portal', element);
|
|
88
|
+
}
|
|
89
|
+
// handle raw value
|
|
90
|
+
if (name === SPECIAL_DOM_TAG.NGT_VALUE) {
|
|
91
|
+
return this.store.createNode('three', Object.assign({ __ngt_renderer__: { rawValue: undefined } }, { __ngt__: { isRaw: true } }));
|
|
92
|
+
}
|
|
93
|
+
const { injectedArgs, injectedParent, store } = this.store.getCreationState();
|
|
94
|
+
let parent = injectedParent;
|
|
95
|
+
if (typeof injectedParent === 'string') {
|
|
96
|
+
parent = store
|
|
97
|
+
.get('scene')
|
|
98
|
+
.getObjectByName(injectedParent);
|
|
99
|
+
}
|
|
100
|
+
// handle primitive
|
|
101
|
+
if (name === SPECIAL_DOM_TAG.NGT_PRIMITIVE) {
|
|
102
|
+
if (!injectedArgs[0])
|
|
103
|
+
throw new Error(`[NGT] ngt-primitive without args is invalid`);
|
|
104
|
+
const object = injectedArgs[0];
|
|
105
|
+
let localState = getLocalState(object);
|
|
106
|
+
if (!Object.keys(localState).length) {
|
|
107
|
+
prepare(object, { store, args: injectedArgs, primitive: true });
|
|
108
|
+
localState = getLocalState(object);
|
|
109
|
+
}
|
|
110
|
+
if (!localState.store)
|
|
111
|
+
localState.store = store;
|
|
112
|
+
const node = this.store.createNode('three', object);
|
|
113
|
+
if (parent) {
|
|
114
|
+
node.__ngt_renderer__[2 /* NgtRendererClassId.injectedParent */] = parent;
|
|
115
|
+
}
|
|
116
|
+
return node;
|
|
117
|
+
}
|
|
118
|
+
const threeTag = name.startsWith('ngt') ? name.slice(4) : name;
|
|
119
|
+
const threeName = kebabToPascal(threeTag);
|
|
120
|
+
const threeTarget = this.catalogue[threeName];
|
|
121
|
+
// we have the THREE constructor here, handle it
|
|
122
|
+
if (threeTarget) {
|
|
123
|
+
const instance = prepare(new threeTarget(...injectedArgs), { store, args: injectedArgs });
|
|
124
|
+
const node = this.store.createNode('three', instance);
|
|
125
|
+
const localState = getLocalState(instance);
|
|
126
|
+
// auto-attach for geometry and material
|
|
127
|
+
if (is.geometry(instance)) {
|
|
128
|
+
localState.attach = ['geometry'];
|
|
129
|
+
}
|
|
130
|
+
else if (is.material(instance)) {
|
|
131
|
+
localState.attach = ['material'];
|
|
132
|
+
}
|
|
133
|
+
if (parent) {
|
|
134
|
+
node.__ngt_renderer__[2 /* NgtRendererClassId.injectedParent */] = parent;
|
|
135
|
+
}
|
|
136
|
+
return node;
|
|
137
|
+
}
|
|
138
|
+
return this.store.createNode('dom', element);
|
|
139
|
+
}
|
|
140
|
+
createComment(value) {
|
|
141
|
+
return this.store.createNode('comment', this.delegate.createComment(value));
|
|
142
|
+
}
|
|
143
|
+
appendChild(parent, newChild) {
|
|
144
|
+
// TODO: just ignore text node for now
|
|
145
|
+
if (newChild instanceof Text)
|
|
146
|
+
return;
|
|
147
|
+
const cRS = newChild.__ngt_renderer__;
|
|
148
|
+
const pRS = parent.__ngt_renderer__;
|
|
149
|
+
if (cRS[0 /* NgtRendererClassId.type */] === 'comment') {
|
|
150
|
+
this.store.setParent(newChild, parent);
|
|
151
|
+
return;
|
|
152
|
+
}
|
|
153
|
+
if (cRS[2 /* NgtRendererClassId.injectedParent */]) {
|
|
154
|
+
if (is.ref(cRS[2 /* NgtRendererClassId.injectedParent */])) {
|
|
155
|
+
cRS[2 /* NgtRendererClassId.injectedParent */].$.pipe(take(1)).subscribe((val) => {
|
|
156
|
+
if (val !== parent) {
|
|
157
|
+
this.appendChild(val, newChild);
|
|
158
|
+
}
|
|
159
|
+
});
|
|
160
|
+
return;
|
|
161
|
+
}
|
|
162
|
+
else if (parent !== cRS[2 /* NgtRendererClassId.injectedParent */]) {
|
|
163
|
+
this.appendChild(cRS[2 /* NgtRendererClassId.injectedParent */], newChild);
|
|
164
|
+
return;
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
this.store.setParent(newChild, parent);
|
|
168
|
+
this.store.addChild(parent, newChild);
|
|
169
|
+
// if new child is a portal
|
|
170
|
+
if (cRS[0 /* NgtRendererClassId.type */] === 'portal') {
|
|
171
|
+
this.store.processPortalContainer(newChild);
|
|
172
|
+
if (cRS[13 /* NgtRendererClassId.portalContainer */]) {
|
|
173
|
+
this.appendChild(parent, cRS[13 /* NgtRendererClassId.portalContainer */]);
|
|
174
|
+
}
|
|
175
|
+
return;
|
|
176
|
+
}
|
|
177
|
+
// if parent is a portal
|
|
178
|
+
if (pRS[0 /* NgtRendererClassId.type */] === 'portal') {
|
|
179
|
+
this.store.processPortalContainer(parent);
|
|
180
|
+
if (pRS[13 /* NgtRendererClassId.portalContainer */]) {
|
|
181
|
+
this.appendChild(pRS[13 /* NgtRendererClassId.portalContainer */], newChild);
|
|
182
|
+
}
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
// if both are three instances, straightforward case
|
|
186
|
+
if (pRS[0 /* NgtRendererClassId.type */] === 'three' && cRS[0 /* NgtRendererClassId.type */] === 'three') {
|
|
187
|
+
// if child already attached to a parent, skip
|
|
188
|
+
if (getLocalState(newChild).parent)
|
|
189
|
+
return;
|
|
190
|
+
// attach THREE child
|
|
191
|
+
attachThreeChild(parent, newChild);
|
|
192
|
+
// here, we handle the special case of if the parent has a compoundParent, which means this child is part of a compound parent template
|
|
193
|
+
if (!cRS[5 /* NgtRendererClassId.compound */])
|
|
194
|
+
return;
|
|
195
|
+
const closestGrandparentWithCompound = this.store.getClosestParentWithCompound(parent);
|
|
196
|
+
if (!closestGrandparentWithCompound)
|
|
197
|
+
return;
|
|
198
|
+
this.appendChild(closestGrandparentWithCompound, newChild);
|
|
199
|
+
return;
|
|
200
|
+
}
|
|
201
|
+
// if only the parent is the THREE instance
|
|
202
|
+
if (pRS[0 /* NgtRendererClassId.type */] === 'three') {
|
|
203
|
+
for (const renderChild of cRS[3 /* NgtRendererClassId.children */]) {
|
|
204
|
+
this.appendChild(parent, renderChild);
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
// if parent is a compound
|
|
208
|
+
if (pRS[0 /* NgtRendererClassId.type */] === 'compound') {
|
|
209
|
+
// if compound doesn't have a THREE instance set yet
|
|
210
|
+
if (!pRS[7 /* NgtRendererClassId.compounded */] && cRS[0 /* NgtRendererClassId.type */] === 'three') {
|
|
211
|
+
// if child is indeed an ngtCompound
|
|
212
|
+
if (cRS[5 /* NgtRendererClassId.compound */])
|
|
213
|
+
this.store.setCompound(parent, newChild);
|
|
214
|
+
// if not, we track the parent (that is supposedly the compound component) on this three instance
|
|
215
|
+
else if (!cRS[6 /* NgtRendererClassId.compoundParent */])
|
|
216
|
+
cRS[6 /* NgtRendererClassId.compoundParent */] = parent;
|
|
217
|
+
}
|
|
218
|
+
// reset the compound if it's changed
|
|
219
|
+
if (pRS[7 /* NgtRendererClassId.compounded */] &&
|
|
220
|
+
cRS[0 /* NgtRendererClassId.type */] === 'three' &&
|
|
221
|
+
cRS[5 /* NgtRendererClassId.compound */] &&
|
|
222
|
+
pRS[7 /* NgtRendererClassId.compounded */] !== newChild) {
|
|
223
|
+
this.store.setCompound(parent, newChild);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
const shouldFindGrandparentInstance =
|
|
227
|
+
// if child is three but haven't been attached to a parent yet
|
|
228
|
+
(cRS[0 /* NgtRendererClassId.type */] === 'three' && !getLocalState(newChild).parent) ||
|
|
229
|
+
// or both parent and child are DOM elements
|
|
230
|
+
// or they are compound AND haven't had a THREE instance yet
|
|
231
|
+
((pRS[0 /* NgtRendererClassId.type */] === 'dom' ||
|
|
232
|
+
(pRS[0 /* NgtRendererClassId.type */] === 'compound' && !pRS[7 /* NgtRendererClassId.compounded */])) &&
|
|
233
|
+
(cRS[0 /* NgtRendererClassId.type */] === 'dom' ||
|
|
234
|
+
(cRS[0 /* NgtRendererClassId.type */] === 'compound' && !cRS[7 /* NgtRendererClassId.compounded */])));
|
|
235
|
+
if (shouldFindGrandparentInstance) {
|
|
236
|
+
// we'll try to get the grandparent instance here so that we can run appendChild with both instances
|
|
237
|
+
const closestGrandparentInstance = this.store.getClosestParentWithInstance(parent);
|
|
238
|
+
if (closestGrandparentInstance)
|
|
239
|
+
this.appendChild(closestGrandparentInstance, newChild);
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
insertBefore(parent, newChild
|
|
243
|
+
// TODO we might need these?
|
|
244
|
+
// refChild: NgtRendererNode
|
|
245
|
+
// isMove?: boolean | undefined
|
|
246
|
+
) {
|
|
247
|
+
if (parent == null || !parent.__ngt_renderer__ || parent === newChild)
|
|
248
|
+
return;
|
|
249
|
+
this.appendChild(parent, newChild);
|
|
250
|
+
}
|
|
251
|
+
removeChild(parent, oldChild, isHostElement) {
|
|
252
|
+
const pRS = parent.__ngt_renderer__;
|
|
253
|
+
const cRS = oldChild.__ngt_renderer__;
|
|
254
|
+
if (pRS[0 /* NgtRendererClassId.type */] === 'three' && cRS[0 /* NgtRendererClassId.type */] === 'three') {
|
|
255
|
+
removeThreeChild(parent, oldChild, true);
|
|
256
|
+
this.store.destroy(oldChild, parent);
|
|
257
|
+
return;
|
|
258
|
+
}
|
|
259
|
+
if (pRS[0 /* NgtRendererClassId.type */] === 'compound' && pRS[1 /* NgtRendererClassId.parent */]) {
|
|
260
|
+
this.removeChild(pRS[1 /* NgtRendererClassId.parent */], oldChild, isHostElement);
|
|
261
|
+
return;
|
|
262
|
+
}
|
|
263
|
+
if (pRS[0 /* NgtRendererClassId.type */] === 'three') {
|
|
264
|
+
this.store.destroy(oldChild, parent);
|
|
265
|
+
return;
|
|
266
|
+
}
|
|
267
|
+
const closestGrandparentInstance = this.store.getClosestParentWithInstance(parent);
|
|
268
|
+
if (closestGrandparentInstance)
|
|
269
|
+
this.removeChild(closestGrandparentInstance, oldChild, isHostElement);
|
|
270
|
+
this.store.destroy(oldChild, closestGrandparentInstance);
|
|
271
|
+
}
|
|
272
|
+
parentNode(node) {
|
|
273
|
+
const rS = node.__ngt_renderer__;
|
|
274
|
+
if (rS?.[1 /* NgtRendererClassId.parent */])
|
|
275
|
+
return rS[1 /* NgtRendererClassId.parent */];
|
|
276
|
+
return this.delegate.parentNode(node);
|
|
277
|
+
}
|
|
278
|
+
setAttribute(el, name, value, namespace) {
|
|
279
|
+
const rS = el.__ngt_renderer__;
|
|
280
|
+
if (rS[0 /* NgtRendererClassId.type */] === 'compound') {
|
|
281
|
+
// we don't have the compound instance yet
|
|
282
|
+
rS[9 /* NgtRendererClassId.attributes */][name] = value;
|
|
283
|
+
if (!rS[7 /* NgtRendererClassId.compounded */]) {
|
|
284
|
+
this.store.queueOperation(el, ['op', () => this.setAttribute(el, name, value, namespace)]);
|
|
285
|
+
return;
|
|
286
|
+
}
|
|
287
|
+
this.setAttribute(rS[7 /* NgtRendererClassId.compounded */], name, value, namespace);
|
|
288
|
+
return;
|
|
289
|
+
}
|
|
290
|
+
if (rS[0 /* NgtRendererClassId.type */] === 'three')
|
|
291
|
+
this.store.applyAttribute(el, name, value);
|
|
292
|
+
}
|
|
293
|
+
setProperty(el, name, value) {
|
|
294
|
+
const rS = el.__ngt_renderer__;
|
|
295
|
+
if (rS[0 /* NgtRendererClassId.type */] === 'compound') {
|
|
296
|
+
// we don't have the compound instance yet
|
|
297
|
+
rS[10 /* NgtRendererClassId.properties */][name] = value;
|
|
298
|
+
if (!rS[7 /* NgtRendererClassId.compounded */]) {
|
|
299
|
+
this.store.queueOperation(el, ['op', () => this.setProperty(el, name, value)]);
|
|
300
|
+
return;
|
|
301
|
+
}
|
|
302
|
+
if (rS[7 /* NgtRendererClassId.compounded */].__ngt_renderer__[5 /* NgtRendererClassId.compound */]) {
|
|
303
|
+
Object.assign(rS[7 /* NgtRendererClassId.compounded */].__ngt_renderer__[5 /* NgtRendererClassId.compound */], {
|
|
304
|
+
props: Object.assign(rS[7 /* NgtRendererClassId.compounded */].__ngt_renderer__[5 /* NgtRendererClassId.compound */], { [name]: value }),
|
|
305
|
+
});
|
|
306
|
+
}
|
|
307
|
+
this.setProperty(rS[7 /* NgtRendererClassId.compounded */], name, value);
|
|
308
|
+
return;
|
|
309
|
+
}
|
|
310
|
+
if (rS[0 /* NgtRendererClassId.type */] === 'three')
|
|
311
|
+
this.store.applyProperty(el, name, value);
|
|
312
|
+
}
|
|
313
|
+
listen(target, eventName, callback) {
|
|
314
|
+
const rS = target.__ngt_renderer__;
|
|
315
|
+
const targetCdr = rS?.[14 /* NgtRendererClassId.injectorFactory */]?.().get(ChangeDetectorRef, null);
|
|
316
|
+
if (rS[0 /* NgtRendererClassId.type */] === 'three' ||
|
|
317
|
+
(rS[0 /* NgtRendererClassId.type */] === 'compound' && rS[7 /* NgtRendererClassId.compounded */])) {
|
|
318
|
+
const instance = rS[7 /* NgtRendererClassId.compounded */] || target;
|
|
319
|
+
const priority = getLocalState(target).priority;
|
|
320
|
+
return processThreeEvent(instance, priority || 0, eventName, callback, this.store.rootCdr, targetCdr);
|
|
321
|
+
}
|
|
322
|
+
if (rS[0 /* NgtRendererClassId.type */] === 'compound' && !rS[7 /* NgtRendererClassId.compounded */]) {
|
|
323
|
+
this.store.queueOperation(target, [
|
|
324
|
+
'op',
|
|
325
|
+
() => this.store.queueOperation(target, ['cleanUp', this.listen(target, eventName, callback)]),
|
|
326
|
+
]);
|
|
327
|
+
}
|
|
328
|
+
// setup a new callback with CDR so that it will trigger change detection properly
|
|
329
|
+
const callbackWithCdr = (event) => {
|
|
330
|
+
const value = callback(event);
|
|
331
|
+
safeDetectChanges(targetCdr);
|
|
332
|
+
safeDetectChanges(this.store.rootCdr);
|
|
333
|
+
return value;
|
|
334
|
+
};
|
|
335
|
+
// if the target doesn't have __ngt_renderer__, we delegate
|
|
336
|
+
if (!rS) {
|
|
337
|
+
return this.delegate.listen(target, eventName, callbackWithCdr);
|
|
338
|
+
}
|
|
339
|
+
// if target is DOM node, then we pass that to delegate Renderer
|
|
340
|
+
if (this.store.isDOM(target)) {
|
|
341
|
+
return this.delegate.listen(target, eventName, callbackWithCdr);
|
|
342
|
+
}
|
|
343
|
+
// @ts-expect-error - we know that target is not DOM node
|
|
344
|
+
if (target === this.store.rootScene) {
|
|
345
|
+
let [domTarget, event] = eventName.split(':');
|
|
346
|
+
if (event == null) {
|
|
347
|
+
event = domTarget;
|
|
348
|
+
domTarget = '';
|
|
349
|
+
}
|
|
350
|
+
const eventTarget = domTarget === 'window'
|
|
351
|
+
? target['ownerDocument']['defaultView']
|
|
352
|
+
: target['ownerDocument'];
|
|
353
|
+
return this.delegate.listen(eventTarget, event, callbackWithCdr);
|
|
354
|
+
}
|
|
355
|
+
return () => { };
|
|
356
|
+
}
|
|
357
|
+
get data() {
|
|
358
|
+
return this.delegate.data;
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"renderer.js","sourceRoot":"","sources":["../../../../../../libs/angular-three/src/lib/renderer/renderer.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n    ChangeDetectorRef,\n    getDebugNode,\n    inject,\n    Injectable,\n    Renderer2,\n    RendererFactory2,\n    RendererType2,\n} from '@angular/core';\nimport { take } from 'rxjs';\nimport { NGT_CATALOGUE } from '../di/catalogue';\nimport { NgtStore } from '../stores/store';\nimport { NgtAnyRecord } from '../types';\nimport { getLocalState, prepare } from '../utils/instance';\nimport { is } from '../utils/is';\nimport { safeDetectChanges } from '../utils/safe-detect-changes';\nimport { NGT_COMPOUND_PREFIXES } from './di';\nimport { NgtRendererClassId } from './enums';\nimport { NgtRendererNode, NgtRendererState, NgtRendererStore } from './store';\nimport { attachThreeChild, kebabToPascal, processThreeEvent, removeThreeChild, SPECIAL_DOM_TAG } from './utils';\n\n@Injectable()\nexport class NgtRendererFactory implements RendererFactory2 {\n    private readonly delegateRendererFactory = inject(RendererFactory2, { skipSelf: true });\n    private readonly catalogue = inject(NGT_CATALOGUE);\n\n    private readonly rendererMap = new Map<string, Renderer2>();\n    private readonly routedSet = new Set<string>();\n    // all Renderer instances share the same Store\n    private readonly rendererStore = new NgtRendererStore({\n        store: inject(NgtStore),\n        cdr: inject(ChangeDetectorRef),\n        portals: [],\n        compoundPrefixes: inject(NGT_COMPOUND_PREFIXES),\n        document: inject(DOCUMENT),\n    });\n\n    createRenderer(hostElement: any, type: RendererType2 | null): Renderer2 {\n        const delegateRenderer = this.delegateRendererFactory.createRenderer(hostElement, type);\n        if (!type) return delegateRenderer;\n        if ((type as NgtAnyRecord)['type']['isRoutedScene']) {\n            this.routedSet.add(type.id);\n        }\n        let renderer = this.rendererMap.get(type.id);\n        if (!renderer) {\n            renderer = new NgtRenderer(\n                delegateRenderer,\n                this.rendererStore,\n                this.catalogue,\n                // setting root scene if there's no routed scene OR this component is the routed Scene\n                !hostElement && (this.routedSet.size === 0 || this.routedSet.has(type.id))\n            );\n            this.rendererMap.set(type.id, renderer);\n        }\n        return renderer;\n    }\n}\n\n/**\n * Anything abbreviated with rS/RS stands for RendererState\n */\nexport class NgtRenderer implements Renderer2 {\n    constructor(\n        private readonly delegate: Renderer2,\n        private readonly store: NgtRendererStore,\n        private readonly catalogue: Record<string, new (...args: any[]) => any>,\n        private root = true\n    ) {}\n\n    createElement(name: string, namespace?: string | null | undefined) {\n        const element = this.delegate.createElement(name, namespace);\n\n        // on first pass, we return the Root Scene as the root node\n        if (this.root) {\n            this.root = false;\n            const node = this.store.createNode('three', this.store.rootScene);\n            node.__ngt_renderer__[NgtRendererClassId.injectorFactory] = () => getDebugNode(element)!.injector;\n            return node;\n        }\n\n        // handle compound\n        if (this.store.isCompound(name)) {\n            return this.store.createNode('compound', element);\n        }\n\n        // handle portal\n        if (name === SPECIAL_DOM_TAG.NGT_PORTAL) {\n            return this.store.createNode('portal', element);\n        }\n\n        // handle raw value\n        if (name === SPECIAL_DOM_TAG.NGT_VALUE) {\n            return this.store.createNode(\n                'three',\n                Object.assign({ __ngt_renderer__: { rawValue: undefined } }, { __ngt__: { isRaw: true } })\n            );\n        }\n\n        const { injectedArgs, injectedParent, store } = this.store.getCreationState();\n\n        let parent = injectedParent as NgtRendererState[NgtRendererClassId.injectedParent];\n        if (typeof injectedParent === 'string') {\n            parent = store\n                .get('scene')\n                .getObjectByName(injectedParent) as unknown as NgtRendererState[NgtRendererClassId.injectedParent];\n        }\n\n        // handle primitive\n        if (name === SPECIAL_DOM_TAG.NGT_PRIMITIVE) {\n            if (!injectedArgs[0]) throw new Error(`[NGT] ngt-primitive without args is invalid`);\n            const object = injectedArgs[0];\n            let localState = getLocalState(object);\n            if (!Object.keys(localState).length) {\n                prepare(object, { store, args: injectedArgs, primitive: true });\n                localState = getLocalState(object);\n            }\n            if (!localState.store) localState.store = store;\n            const node = this.store.createNode('three', object);\n            if (parent) {\n                node.__ngt_renderer__[NgtRendererClassId.injectedParent] = parent;\n            }\n            return node;\n        }\n\n        const threeTag = name.startsWith('ngt') ? name.slice(4) : name;\n        const threeName = kebabToPascal(threeTag);\n        const threeTarget = this.catalogue[threeName];\n        // we have the THREE constructor here, handle it\n        if (threeTarget) {\n            const instance = prepare(new threeTarget(...injectedArgs), { store, args: injectedArgs });\n            const node = this.store.createNode('three', instance);\n            const localState = getLocalState(instance);\n\n            // auto-attach for geometry and material\n            if (is.geometry(instance)) {\n                localState.attach = ['geometry'];\n            } else if (is.material(instance)) {\n                localState.attach = ['material'];\n            }\n\n            if (parent) {\n                node.__ngt_renderer__[NgtRendererClassId.injectedParent] = parent;\n            }\n\n            return node;\n        }\n\n        return this.store.createNode('dom', element);\n    }\n\n    createComment(value: string) {\n        return this.store.createNode('comment', this.delegate.createComment(value));\n    }\n\n    appendChild(parent: NgtRendererNode, newChild: NgtRendererNode): void {\n        // TODO: just ignore text node for now\n        if (newChild instanceof Text) return;\n        const cRS = newChild.__ngt_renderer__;\n        const pRS = parent.__ngt_renderer__;\n\n        if (cRS[NgtRendererClassId.type] === 'comment') {\n            this.store.setParent(newChild, parent);\n            return;\n        }\n\n        if (cRS[NgtRendererClassId.injectedParent]) {\n            if (is.ref(cRS[NgtRendererClassId.injectedParent])) {\n                cRS[NgtRendererClassId.injectedParent].$.pipe(take(1)).subscribe((val) => {\n                    if (val !== parent) {\n                        this.appendChild(val, newChild);\n                    }\n                });\n                return;\n            } else if (parent !== cRS[NgtRendererClassId.injectedParent]) {\n                this.appendChild(cRS[NgtRendererClassId.injectedParent], newChild);\n                return;\n            }\n        }\n\n        this.store.setParent(newChild, parent);\n        this.store.addChild(parent, newChild);\n\n        // if new child is a portal\n        if (cRS[NgtRendererClassId.type] === 'portal') {\n            this.store.processPortalContainer(newChild);\n            if (cRS[NgtRendererClassId.portalContainer]) {\n                this.appendChild(parent, cRS[NgtRendererClassId.portalContainer]);\n            }\n            return;\n        }\n\n        // if parent is a portal\n        if (pRS[NgtRendererClassId.type] === 'portal') {\n            this.store.processPortalContainer(parent);\n            if (pRS[NgtRendererClassId.portalContainer]) {\n                this.appendChild(pRS[NgtRendererClassId.portalContainer], newChild);\n            }\n            return;\n        }\n\n        // if both are three instances, straightforward case\n        if (pRS[NgtRendererClassId.type] === 'three' && cRS[NgtRendererClassId.type] === 'three') {\n            // if child already attached to a parent, skip\n            if (getLocalState(newChild).parent) return;\n            // attach THREE child\n            attachThreeChild(parent, newChild);\n            // here, we handle the special case of if the parent has a compoundParent, which means this child is part of a compound parent template\n            if (!cRS[NgtRendererClassId.compound]) return;\n            const closestGrandparentWithCompound = this.store.getClosestParentWithCompound(parent);\n            if (!closestGrandparentWithCompound) return;\n            this.appendChild(closestGrandparentWithCompound, newChild);\n            return;\n        }\n\n        // if only the parent is the THREE instance\n        if (pRS[NgtRendererClassId.type] === 'three') {\n            for (const renderChild of cRS[NgtRendererClassId.children]) {\n                this.appendChild(parent, renderChild);\n            }\n        }\n\n        // if parent is a compound\n        if (pRS[NgtRendererClassId.type] === 'compound') {\n            // if compound doesn't have a THREE instance set yet\n            if (!pRS[NgtRendererClassId.compounded] && cRS[NgtRendererClassId.type] === 'three') {\n                // if child is indeed an ngtCompound\n                if (cRS[NgtRendererClassId.compound]) this.store.setCompound(parent, newChild);\n                // if not, we track the parent (that is supposedly the compound component) on this three instance\n                else if (!cRS[NgtRendererClassId.compoundParent]) cRS[NgtRendererClassId.compoundParent] = parent;\n            }\n\n            // reset the compound if it's changed\n            if (\n                pRS[NgtRendererClassId.compounded] &&\n                cRS[NgtRendererClassId.type] === 'three' &&\n                cRS[NgtRendererClassId.compound] &&\n                pRS[NgtRendererClassId.compounded] !== newChild\n            ) {\n                this.store.setCompound(parent, newChild);\n            }\n        }\n\n        const shouldFindGrandparentInstance =\n            // if child is three but haven't been attached to a parent yet\n            (cRS[NgtRendererClassId.type] === 'three' && !getLocalState(newChild).parent) ||\n            // or both parent and child are DOM elements\n            // or they are compound AND haven't had a THREE instance yet\n            ((pRS[NgtRendererClassId.type] === 'dom' ||\n                (pRS[NgtRendererClassId.type] === 'compound' && !pRS[NgtRendererClassId.compounded])) &&\n                (cRS[NgtRendererClassId.type] === 'dom' ||\n                    (cRS[NgtRendererClassId.type] === 'compound' && !cRS[NgtRendererClassId.compounded])));\n\n        if (shouldFindGrandparentInstance) {\n            // we'll try to get the grandparent instance here so that we can run appendChild with both instances\n            const closestGrandparentInstance = this.store.getClosestParentWithInstance(parent);\n            if (closestGrandparentInstance) this.appendChild(closestGrandparentInstance, newChild);\n        }\n    }\n\n    insertBefore(\n        parent: NgtRendererNode,\n        newChild: NgtRendererNode\n        // TODO  we might need these?\n        // refChild: NgtRendererNode\n        // isMove?: boolean | undefined\n    ): void {\n        if (parent == null || !parent.__ngt_renderer__ || parent === newChild) return;\n        this.appendChild(parent, newChild);\n    }\n\n    removeChild(parent: NgtRendererNode, oldChild: NgtRendererNode, isHostElement?: boolean | undefined): void {\n        const pRS = parent.__ngt_renderer__;\n        const cRS = oldChild.__ngt_renderer__;\n        if (pRS[NgtRendererClassId.type] === 'three' && cRS[NgtRendererClassId.type] === 'three') {\n            removeThreeChild(parent, oldChild, true);\n            this.store.destroy(oldChild, parent);\n            return;\n        }\n\n        if (pRS[NgtRendererClassId.type] === 'compound' && pRS[NgtRendererClassId.parent]) {\n            this.removeChild(pRS[NgtRendererClassId.parent], oldChild, isHostElement);\n            return;\n        }\n\n        if (pRS[NgtRendererClassId.type] === 'three') {\n            this.store.destroy(oldChild, parent);\n            return;\n        }\n\n        const closestGrandparentInstance = this.store.getClosestParentWithInstance(parent);\n        if (closestGrandparentInstance) this.removeChild(closestGrandparentInstance, oldChild, isHostElement);\n        this.store.destroy(oldChild, closestGrandparentInstance as NgtRendererNode);\n    }\n\n    parentNode(node: NgtRendererNode) {\n        const rS = node.__ngt_renderer__;\n        if (rS?.[NgtRendererClassId.parent]) return rS[NgtRendererClassId.parent];\n        return this.delegate.parentNode(node);\n    }\n\n    setAttribute(el: NgtRendererNode, name: string, value: string, namespace?: string | null | undefined): void {\n        const rS = el.__ngt_renderer__;\n        if (rS[NgtRendererClassId.type] === 'compound') {\n            // we don't have the compound instance yet\n            rS[NgtRendererClassId.attributes][name] = value;\n            if (!rS[NgtRendererClassId.compounded]) {\n                this.store.queueOperation(el, ['op', () => this.setAttribute(el, name, value, namespace)]);\n                return;\n            }\n\n            this.setAttribute(rS[NgtRendererClassId.compounded], name, value, namespace);\n            return;\n        }\n\n        if (rS[NgtRendererClassId.type] === 'three') this.store.applyAttribute(el, name, value);\n    }\n\n    setProperty(el: NgtRendererNode, name: string, value: any): void {\n        const rS = el.__ngt_renderer__;\n        if (rS[NgtRendererClassId.type] === 'compound') {\n            // we don't have the compound instance yet\n            rS[NgtRendererClassId.properties][name] = value;\n            if (!rS[NgtRendererClassId.compounded]) {\n                this.store.queueOperation(el, ['op', () => this.setProperty(el, name, value)]);\n                return;\n            }\n\n            if (rS[NgtRendererClassId.compounded].__ngt_renderer__[NgtRendererClassId.compound]) {\n                Object.assign(rS[NgtRendererClassId.compounded].__ngt_renderer__[NgtRendererClassId.compound], {\n                    props: Object.assign(\n                        rS[NgtRendererClassId.compounded].__ngt_renderer__[NgtRendererClassId.compound],\n                        { [name]: value }\n                    ),\n                });\n            }\n            this.setProperty(rS[NgtRendererClassId.compounded], name, value);\n            return;\n        }\n\n        if (rS[NgtRendererClassId.type] === 'three') this.store.applyProperty(el, name, value);\n    }\n\n    listen(target: NgtRendererNode, eventName: string, callback: (event: any) => boolean | void): () => void {\n        const rS = target.__ngt_renderer__;\n        const targetCdr = rS?.[NgtRendererClassId.injectorFactory]?.().get(ChangeDetectorRef, null);\n\n        if (\n            rS[NgtRendererClassId.type] === 'three' ||\n            (rS[NgtRendererClassId.type] === 'compound' && rS[NgtRendererClassId.compounded])\n        ) {\n            const instance = rS[NgtRendererClassId.compounded] || target;\n            const priority = getLocalState(target).priority;\n            return processThreeEvent(instance, priority || 0, eventName, callback, this.store.rootCdr, targetCdr!);\n        }\n\n        if (rS[NgtRendererClassId.type] === 'compound' && !rS[NgtRendererClassId.compounded]) {\n            this.store.queueOperation(target, [\n                'op',\n                () => this.store.queueOperation(target, ['cleanUp', this.listen(target, eventName, callback)]),\n            ]);\n        }\n\n        // setup a new callback with CDR so that it will trigger change detection properly\n        const callbackWithCdr = (event: any) => {\n            const value = callback(event);\n            safeDetectChanges(targetCdr);\n            safeDetectChanges(this.store.rootCdr);\n            return value;\n        };\n\n        // if the target doesn't have __ngt_renderer__, we delegate\n        if (!rS) {\n            return this.delegate.listen(target, eventName, callbackWithCdr);\n        }\n\n        // if target is DOM node, then we pass that to delegate Renderer\n        if (this.store.isDOM(target)) {\n            return this.delegate.listen(target, eventName, callbackWithCdr);\n        }\n\n        // @ts-expect-error - we know that target is not DOM node\n        if (target === this.store.rootScene) {\n            let [domTarget, event] = eventName.split(':');\n            if (event == null) {\n                event = domTarget;\n                domTarget = '';\n            }\n            const eventTarget =\n                domTarget === 'window'\n                    ? (target as NgtAnyRecord)['ownerDocument']['defaultView']\n                    : (target as NgtAnyRecord)['ownerDocument'];\n            return this.delegate.listen(eventTarget, event, callbackWithCdr);\n        }\n\n        return () => {};\n    }\n\n    createText = this.delegate.createText.bind(this.delegate);\n    destroy = this.delegate.destroy.bind(this.delegate);\n    destroyNode: ((node: any) => void) | null = null;\n    selectRootElement = this.delegate.selectRootElement.bind(this.delegate);\n    nextSibling = this.delegate.nextSibling.bind(this.delegate);\n    removeAttribute = this.delegate.removeAttribute.bind(this.delegate);\n    addClass = this.delegate.addClass.bind(this.delegate);\n    removeClass = this.delegate.removeClass.bind(this.delegate);\n    setStyle = this.delegate.setStyle.bind(this.delegate);\n    removeStyle = this.delegate.removeStyle.bind(this.delegate);\n    setValue = this.delegate.setValue.bind(this.delegate);\n    get data(): { [key: string]: any } {\n        return this.delegate.data;\n    }\n}\n"]}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { ActivationEnd, Router, RouterOutlet } from '@angular/router';
|
|
3
|
+
import { filter, takeUntil } from 'rxjs';
|
|
4
|
+
import { injectNgtDestroy } from './di/destroy';
|
|
5
|
+
import { safeDetectChanges } from './utils/safe-detect-changes';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/router";
|
|
8
|
+
class NgtRoutedScene {
|
|
9
|
+
static { this.isRoutedScene = true; }
|
|
10
|
+
constructor(router) {
|
|
11
|
+
const { destroy$, cdr } = injectNgtDestroy();
|
|
12
|
+
router.events
|
|
13
|
+
.pipe(filter((event) => event instanceof ActivationEnd), takeUntil(destroy$))
|
|
14
|
+
.subscribe(() => safeDetectChanges(cdr));
|
|
15
|
+
}
|
|
16
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtRoutedScene, deps: [{ token: i1.Router }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
17
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: NgtRoutedScene, isStandalone: true, selector: "ngt-routed-scene", ngImport: i0, template: `<router-outlet />`, isInline: true, dependencies: [{ kind: "directive", type: RouterOutlet, selector: "router-outlet", inputs: ["name"], outputs: ["activate", "deactivate", "attach", "detach"], exportAs: ["outlet"] }] }); }
|
|
18
|
+
}
|
|
19
|
+
export { NgtRoutedScene };
|
|
20
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtRoutedScene, decorators: [{
|
|
21
|
+
type: Component,
|
|
22
|
+
args: [{
|
|
23
|
+
standalone: true,
|
|
24
|
+
selector: 'ngt-routed-scene',
|
|
25
|
+
template: `<router-outlet />`,
|
|
26
|
+
imports: [RouterOutlet],
|
|
27
|
+
}]
|
|
28
|
+
}], ctorParameters: function () { return [{ type: i1.Router }]; } });
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm91dGVkLXNjZW5lLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vbGlicy9hbmd1bGFyLXRocmVlL3NyYy9saWIvcm91dGVkLXNjZW5lLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDMUMsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDdEUsT0FBTyxFQUFFLE1BQU0sRUFBRSxTQUFTLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDekMsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sY0FBYyxDQUFDO0FBQ2hELE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLDZCQUE2QixDQUFDOzs7QUFFaEUsTUFNYSxjQUFjO2FBQ2hCLGtCQUFhLEdBQUcsSUFBSSxBQUFQLENBQVE7SUFFNUIsWUFBWSxNQUFjO1FBQ3RCLE1BQU0sRUFBRSxRQUFRLEVBQUUsR0FBRyxFQUFFLEdBQUcsZ0JBQWdCLEVBQUUsQ0FBQztRQUM3QyxNQUFNLENBQUMsTUFBTTthQUNSLElBQUksQ0FDRCxNQUFNLENBQUMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLEtBQUssWUFBWSxhQUFhLENBQUMsRUFDakQsU0FBUyxDQUFDLFFBQVEsQ0FBQyxDQUN0QjthQUNBLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxpQkFBaUIsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQ2pELENBQUM7OEdBWFEsY0FBYztrR0FBZCxjQUFjLDRFQUhiLG1CQUFtQiw0REFDbkIsWUFBWTs7U0FFYixjQUFjOzJGQUFkLGNBQWM7a0JBTjFCLFNBQVM7bUJBQUM7b0JBQ1AsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLFFBQVEsRUFBRSxrQkFBa0I7b0JBQzVCLFFBQVEsRUFBRSxtQkFBbUI7b0JBQzdCLE9BQU8sRUFBRSxDQUFDLFlBQVksQ0FBQztpQkFDMUIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEFjdGl2YXRpb25FbmQsIFJvdXRlciwgUm91dGVyT3V0bGV0IH0gZnJvbSAnQGFuZ3VsYXIvcm91dGVyJztcbmltcG9ydCB7IGZpbHRlciwgdGFrZVVudGlsIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBpbmplY3ROZ3REZXN0cm95IH0gZnJvbSAnLi9kaS9kZXN0cm95JztcbmltcG9ydCB7IHNhZmVEZXRlY3RDaGFuZ2VzIH0gZnJvbSAnLi91dGlscy9zYWZlLWRldGVjdC1jaGFuZ2VzJztcblxuQENvbXBvbmVudCh7XG4gICAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgICBzZWxlY3RvcjogJ25ndC1yb3V0ZWQtc2NlbmUnLFxuICAgIHRlbXBsYXRlOiBgPHJvdXRlci1vdXRsZXQgLz5gLFxuICAgIGltcG9ydHM6IFtSb3V0ZXJPdXRsZXRdLFxufSlcbmV4cG9ydCBjbGFzcyBOZ3RSb3V0ZWRTY2VuZSB7XG4gICAgc3RhdGljIGlzUm91dGVkU2NlbmUgPSB0cnVlO1xuXG4gICAgY29uc3RydWN0b3Iocm91dGVyOiBSb3V0ZXIpIHtcbiAgICAgICAgY29uc3QgeyBkZXN0cm95JCwgY2RyIH0gPSBpbmplY3ROZ3REZXN0cm95KCk7XG4gICAgICAgIHJvdXRlci5ldmVudHNcbiAgICAgICAgICAgIC5waXBlKFxuICAgICAgICAgICAgICAgIGZpbHRlcigoZXZlbnQpID0+IGV2ZW50IGluc3RhbmNlb2YgQWN0aXZhdGlvbkVuZCksXG4gICAgICAgICAgICAgICAgdGFrZVVudGlsKGRlc3Ryb3kkKVxuICAgICAgICAgICAgKVxuICAgICAgICAgICAgLnN1YnNjcmliZSgoKSA9PiBzYWZlRGV0ZWN0Q2hhbmdlcyhjZHIpKTtcbiAgICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { RxState } from '@rx-angular/state';
|
|
3
|
+
import { combineLatest, startWith, tap } from 'rxjs';
|
|
4
|
+
import { is } from '../utils/is';
|
|
5
|
+
import { safeDetectChanges } from '../utils/safe-detect-changes';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
export const startWithUndefined = () => startWith(undefined);
|
|
8
|
+
/**
|
|
9
|
+
* An extended `tap` operator that accepts an `effectFn` which:
|
|
10
|
+
* - runs on every `next` notification from `source$`
|
|
11
|
+
* - can optionally return a `cleanUp` function that
|
|
12
|
+
* invokes from the 2nd `next` notification onward and on `unsubscribe` (destroyed)
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* ```typescript
|
|
16
|
+
* source$.pipe(
|
|
17
|
+
* tapEffect((sourceValue) = {
|
|
18
|
+
* const cb = () => {
|
|
19
|
+
* doStuff(sourceValue);
|
|
20
|
+
* };
|
|
21
|
+
* addListener('event', cb);
|
|
22
|
+
*
|
|
23
|
+
* return () => {
|
|
24
|
+
* removeListener('event', cb);
|
|
25
|
+
* }
|
|
26
|
+
* })
|
|
27
|
+
* )
|
|
28
|
+
* ```
|
|
29
|
+
*/
|
|
30
|
+
export function tapEffect(effectFn) {
|
|
31
|
+
let cleanupFn = () => { };
|
|
32
|
+
let firstRun = false;
|
|
33
|
+
let prev = undefined;
|
|
34
|
+
const teardown = (error) => () => {
|
|
35
|
+
if (cleanupFn)
|
|
36
|
+
cleanupFn({ prev, complete: true, error });
|
|
37
|
+
};
|
|
38
|
+
return tap({
|
|
39
|
+
next: (value) => {
|
|
40
|
+
if (cleanupFn && firstRun)
|
|
41
|
+
cleanupFn({ prev, complete: false, error: false });
|
|
42
|
+
const cleanUpOrVoid = effectFn(value);
|
|
43
|
+
if (cleanUpOrVoid)
|
|
44
|
+
cleanupFn = cleanUpOrVoid;
|
|
45
|
+
prev = value;
|
|
46
|
+
if (!firstRun)
|
|
47
|
+
firstRun = true;
|
|
48
|
+
},
|
|
49
|
+
complete: teardown(false),
|
|
50
|
+
unsubscribe: teardown(false),
|
|
51
|
+
error: teardown(true),
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
class NgtRxStore extends RxState {
|
|
55
|
+
constructor() {
|
|
56
|
+
super();
|
|
57
|
+
// set a dummy property so that initial this.get() won't return undefined
|
|
58
|
+
this.set({ __ngt_dummy__: '__ngt_dummy__' });
|
|
59
|
+
// override set so our consumers don't have to handle undefined for state that already have default values
|
|
60
|
+
const originalSet = this.set.bind(this);
|
|
61
|
+
Object.defineProperty(this, 'set', {
|
|
62
|
+
get: () => {
|
|
63
|
+
return (...args) => {
|
|
64
|
+
const firstArg = args[0];
|
|
65
|
+
if (is.obj(firstArg)) {
|
|
66
|
+
const modArgs = Object.entries(firstArg).reduce((modded, [key, value]) => {
|
|
67
|
+
modded[key] = value === undefined ? this.get(key) : value;
|
|
68
|
+
return modded;
|
|
69
|
+
}, {});
|
|
70
|
+
return originalSet(modArgs);
|
|
71
|
+
}
|
|
72
|
+
return originalSet(...args);
|
|
73
|
+
};
|
|
74
|
+
},
|
|
75
|
+
});
|
|
76
|
+
// override get to return {} if get() returns undefined
|
|
77
|
+
const originalGet = this.get.bind(this);
|
|
78
|
+
Object.defineProperty(this, 'get', {
|
|
79
|
+
get: () => (...args) => {
|
|
80
|
+
if (args.length === 0)
|
|
81
|
+
return originalGet() ?? {};
|
|
82
|
+
return originalGet(...args);
|
|
83
|
+
},
|
|
84
|
+
});
|
|
85
|
+
// call initialize that might be setup by derived Stores
|
|
86
|
+
this.initialize();
|
|
87
|
+
}
|
|
88
|
+
initialize() {
|
|
89
|
+
return;
|
|
90
|
+
}
|
|
91
|
+
effect(obs, sideEffectFn) {
|
|
92
|
+
return this.hold(obs.pipe(tapEffect(sideEffectFn)));
|
|
93
|
+
}
|
|
94
|
+
triggerChangeDetection(cdr, keys = []) {
|
|
95
|
+
let $ = this.$;
|
|
96
|
+
if (keys.length) {
|
|
97
|
+
$ = combineLatest(keys.map((key) => this.select(key).pipe(startWith(this.get(key) ?? undefined))));
|
|
98
|
+
}
|
|
99
|
+
this.hold($, () => void requestAnimationFrame(() => void safeDetectChanges(cdr)));
|
|
100
|
+
}
|
|
101
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtRxStore, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
102
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtRxStore }); }
|
|
103
|
+
}
|
|
104
|
+
export { NgtRxStore };
|
|
105
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtRxStore, decorators: [{
|
|
106
|
+
type: Injectable
|
|
107
|
+
}], ctorParameters: function () { return []; } });
|
|
108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rx-store.js","sourceRoot":"","sources":["../../../../../../libs/angular-three/src/lib/stores/rx-store.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, Injectable } from '@angular/core';\nimport { RxState } from '@rx-angular/state';\nimport { combineLatest, MonoTypeOperatorFunction, Observable, startWith, tap } from 'rxjs';\nimport type { NgtAnyRecord } from '../types';\nimport { is } from '../utils/is';\nimport { safeDetectChanges } from '../utils/safe-detect-changes';\n\nexport const startWithUndefined = <T>(): MonoTypeOperatorFunction<T> => startWith<T>(undefined! as T);\n\ntype EffectFn<TValue> = (\n    value: TValue\n) => void | undefined | ((cleanUpParams: { prev: TValue | undefined; complete: boolean; error: boolean }) => void);\n\n/**\n * An extended `tap` operator that accepts an `effectFn` which:\n * - runs on every `next` notification from `source$`\n * - can optionally return a `cleanUp` function that\n * invokes from the 2nd `next` notification onward and on `unsubscribe` (destroyed)\n *\n * @example\n * ```typescript\n * source$.pipe(\n *  tapEffect((sourceValue) = {\n *    const cb = () => {\n *      doStuff(sourceValue);\n *    };\n *    addListener('event', cb);\n *\n *    return () => {\n *      removeListener('event', cb);\n *    }\n *  })\n * )\n * ```\n */\nexport function tapEffect<TValue>(effectFn: EffectFn<TValue>): MonoTypeOperatorFunction<TValue> {\n    let cleanupFn: (cleanUpParams: { prev: TValue | undefined; complete: boolean; error: boolean }) => void = () => {};\n    let firstRun = false;\n    let prev: TValue | undefined = undefined;\n\n    const teardown = (error: boolean) => () => {\n        if (cleanupFn) cleanupFn({ prev, complete: true, error });\n    };\n\n    return tap<TValue>({\n        next: (value: TValue) => {\n            if (cleanupFn && firstRun) cleanupFn({ prev, complete: false, error: false });\n\n            const cleanUpOrVoid = effectFn(value);\n            if (cleanUpOrVoid) cleanupFn = cleanUpOrVoid;\n\n            prev = value;\n\n            if (!firstRun) firstRun = true;\n        },\n        complete: teardown(false),\n        unsubscribe: teardown(false),\n        error: teardown(true),\n    });\n}\n\n@Injectable()\nexport class NgtRxStore<\n    TState extends object = any,\n    TRxState extends object = TState & Record<string, any>\n> extends RxState<TRxState> {\n    constructor() {\n        super();\n        // set a dummy property so that initial this.get() won't return undefined\n        this.set({ __ngt_dummy__: '__ngt_dummy__' } as TRxState);\n        // override set so our consumers don't have to handle undefined for state that already have default values\n        const originalSet = this.set.bind(this);\n        Object.defineProperty(this, 'set', {\n            get: () => {\n                return (...args: Parameters<RxState<TRxState>['set']>) => {\n                    const firstArg = args[0];\n                    if (is.obj(firstArg)) {\n                        const modArgs = Object.entries(firstArg).reduce((modded, [key, value]) => {\n                            modded[key] = value === undefined ? this.get(key as keyof TRxState) : value;\n                            return modded;\n                        }, {} as NgtAnyRecord);\n                        return originalSet(modArgs as Partial<TRxState>);\n                    }\n                    return originalSet(...args);\n                };\n            },\n        });\n\n        // override get to return {} if get() returns undefined\n        const originalGet = this.get.bind(this);\n        Object.defineProperty(this, 'get', {\n            get:\n                () =>\n                (...args: Parameters<RxState<TRxState>['get']>) => {\n                    if ((args as any[]).length === 0) return originalGet() ?? {};\n                    return originalGet(...args);\n                },\n        });\n\n        // call initialize that might be setup by derived Stores\n        this.initialize();\n    }\n\n    protected initialize() {\n        return;\n    }\n\n    effect<S>(obs: Observable<S>, sideEffectFn: EffectFn<S>): void {\n        return this.hold(obs.pipe(tapEffect(sideEffectFn)));\n    }\n\n    triggerChangeDetection(cdr: ChangeDetectorRef, keys: Array<keyof TRxState> = []) {\n        let $: Observable<any> = this.$;\n\n        if (keys.length) {\n            $ = combineLatest(keys.map((key) => this.select(key).pipe(startWith(this.get(key) ?? undefined))));\n        }\n\n        this.hold($, () => void requestAnimationFrame(() => void safeDetectChanges(cdr)));\n    }\n}\n"]}
|