angular-three 1.10.2 → 2.0.0-beta.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (62) hide show
  1. package/README.md +89 -5
  2. package/esm2022/index.mjs +5 -7
  3. package/esm2022/lib/canvas.mjs +138 -99
  4. package/esm2022/lib/di/before-render.mjs +7 -10
  5. package/esm2022/lib/di/ref.mjs +38 -59
  6. package/esm2022/lib/directives/args.mjs +5 -8
  7. package/esm2022/lib/directives/common.mjs +20 -13
  8. package/esm2022/lib/directives/parent.mjs +5 -8
  9. package/esm2022/lib/{web → dom}/events.mjs +1 -1
  10. package/esm2022/lib/events.mjs +2 -2
  11. package/esm2022/lib/loader.mjs +58 -44
  12. package/esm2022/lib/loop.mjs +6 -8
  13. package/esm2022/lib/portal.mjs +76 -63
  14. package/esm2022/lib/renderer/provider.mjs +3 -2
  15. package/esm2022/lib/renderer/renderer.mjs +53 -52
  16. package/esm2022/lib/renderer/store.mjs +14 -19
  17. package/esm2022/lib/renderer/utils.mjs +27 -18
  18. package/esm2022/lib/routed-scene.mjs +12 -10
  19. package/esm2022/lib/stores/signal.store.mjs +60 -0
  20. package/esm2022/lib/stores/store.mjs +69 -48
  21. package/esm2022/lib/three-types.mjs +2 -0
  22. package/esm2022/lib/types.mjs +1 -1
  23. package/esm2022/lib/utils/apply-props.mjs +11 -7
  24. package/esm2022/lib/utils/attach.mjs +1 -1
  25. package/esm2022/lib/utils/instance.mjs +14 -14
  26. package/esm2022/lib/utils/safe-detect-changes.mjs +1 -1
  27. package/fesm2022/angular-three.mjs +1673 -1744
  28. package/fesm2022/angular-three.mjs.map +1 -1
  29. package/index.d.ts +4 -6
  30. package/lib/canvas.d.ts +11 -20
  31. package/lib/di/before-render.d.ts +5 -1
  32. package/lib/di/ref.d.ts +5 -11
  33. package/lib/directives/args.d.ts +6 -6
  34. package/lib/directives/common.d.ts +1 -4
  35. package/lib/directives/parent.d.ts +1 -1
  36. package/lib/{web → dom}/events.d.ts +2 -2
  37. package/lib/events.d.ts +3 -3
  38. package/lib/loader.d.ts +6 -2
  39. package/lib/loop.d.ts +4 -4
  40. package/lib/portal.d.ts +11 -18
  41. package/lib/renderer/renderer.d.ts +7 -9
  42. package/lib/renderer/store.d.ts +3 -5
  43. package/lib/renderer/utils.d.ts +6 -4
  44. package/lib/routed-scene.d.ts +4 -2
  45. package/lib/stores/signal.store.d.ts +19 -0
  46. package/lib/stores/store.d.ts +4 -7
  47. package/lib/three-types.d.ts +306 -0
  48. package/lib/types.d.ts +22 -25
  49. package/lib/utils/attach.d.ts +2 -2
  50. package/lib/utils/instance.d.ts +1 -1
  51. package/package.json +6 -7
  52. package/plugin/package.json +1 -1
  53. package/plugin/src/generators/init/init.d.ts +1 -1
  54. package/plugin/src/generators/init/init.js +1 -1
  55. package/esm2022/lib/di/destroy.mjs +0 -23
  56. package/esm2022/lib/di/run-in-context.mjs +0 -40
  57. package/esm2022/lib/pipes/push.mjs +0 -50
  58. package/esm2022/lib/stores/rx-store.mjs +0 -108
  59. package/lib/di/destroy.d.ts +0 -9
  60. package/lib/di/run-in-context.d.ts +0 -6
  61. package/lib/pipes/push.d.ts +0 -16
  62. package/lib/stores/rx-store.d.ts +0 -42
@@ -1,55 +1,69 @@
1
- import { ChangeDetectorRef, inject } from '@angular/core';
2
- import { ReplaySubject, catchError, forkJoin, from, isObservable, map, of, retry, share, switchMap, take, tap, } from 'rxjs';
1
+ import { ChangeDetectorRef, Injector, assertInInjectionContext, computed, effect, inject, runInInjectionContext, signal, untracked, } from '@angular/core';
3
2
  import { makeObjectGraph } from './utils/make';
4
3
  import { safeDetectChanges } from './utils/safe-detect-changes';
5
4
  const cached = new Map();
6
- function load(loaderConstructorFactory, input, extensions, onProgress) {
7
- const urls$ = isObservable(input) ? input : of(input);
8
- return urls$.pipe(map((inputs) => {
9
- const loaderConstructor = loaderConstructorFactory(inputs);
5
+ function load(loaderConstructorFactory, inputs, { extensions, onProgress, } = {}) {
6
+ const computedUrls = computed(() => {
7
+ const input = inputs();
8
+ if (Array.isArray(input))
9
+ return input;
10
+ if (typeof input === 'string')
11
+ return [input];
12
+ return Object.values(input);
13
+ });
14
+ return () => {
15
+ const urls = computedUrls();
16
+ const loaderConstructor = loaderConstructorFactory(urls);
10
17
  const loader = new loaderConstructor();
11
18
  if (extensions)
12
19
  extensions(loader);
13
- const urls = Array.isArray(inputs) ? inputs : typeof inputs === 'string' ? [inputs] : Object.values(inputs);
14
- return [
15
- urls.map((url) => {
16
- if (!cached.has(url)) {
17
- cached.set(url, from(loader.loadAsync(url, onProgress)).pipe(tap((data) => {
18
- if (data['scene'])
19
- Object.assign(data, makeObjectGraph(data['scene']));
20
- }), retry(2), catchError((err) => {
21
- console.error(`[NGT] Error loading ${url}: ${err.message}`);
22
- return of([]);
23
- }), share({ connector: () => new ReplaySubject(1) })));
24
- }
25
- return cached.get(url);
26
- }),
27
- inputs,
28
- ];
29
- }));
30
- }
31
- export function injectNgtLoader(loaderConstructorFactory, input, extensions, onProgress) {
32
- const cdr = inject(ChangeDetectorRef);
33
- return load(loaderConstructorFactory, input, extensions, onProgress).pipe(switchMap(([observables$, inputs]) => {
34
- return forkJoin(observables$).pipe(map((results) => {
35
- if (Array.isArray(inputs))
36
- return results;
37
- if (typeof inputs === 'string')
38
- return results[0];
39
- const keys = Object.keys(inputs);
40
- return keys.reduce((result, key) => {
41
- result[key] = results[keys.indexOf(key)];
42
- return result;
43
- }, {});
44
- }), tap(() => {
45
- requestAnimationFrame(() => void safeDetectChanges(cdr));
20
+ return urls.map((url) => new Promise((resolve, reject) => {
21
+ if (cached.has(url)) {
22
+ resolve(cached.get(url));
23
+ }
24
+ else {
25
+ loader.load(url, (data) => {
26
+ if ('scene' in data)
27
+ Object.assign(data, makeObjectGraph(data['scene']));
28
+ cached.set(url, data);
29
+ resolve(data);
30
+ }, onProgress, (error) => reject(new Error(`[NGT] Could not load ${url}: ${error}`)));
31
+ }
46
32
  }));
47
- }));
33
+ };
34
+ }
35
+ export function injectNgtLoader(loaderConstructorFactory, inputs, { extensions, onProgress, injector = inject(Injector, { optional: true }), } = {}) {
36
+ !injector && assertInInjectionContext(injectNgtLoader);
37
+ return runInInjectionContext(injector, () => {
38
+ const cdr = inject(ChangeDetectorRef);
39
+ const response = signal(null);
40
+ const effector = load(loaderConstructorFactory, inputs, { extensions, onProgress });
41
+ effect(() => {
42
+ const originalUrls = untracked(inputs);
43
+ Promise.all(effector())
44
+ .then((results) => {
45
+ if (Array.isArray(originalUrls))
46
+ return results;
47
+ if (typeof originalUrls === 'string')
48
+ return results[0];
49
+ const keys = Object.keys(originalUrls);
50
+ return keys.reduce((result, key) => {
51
+ result[key] = results[keys.indexOf(key)];
52
+ return result;
53
+ }, {});
54
+ })
55
+ .then((value) => {
56
+ response.set(value);
57
+ safeDetectChanges(cdr);
58
+ });
59
+ }, { injector: injector, allowSignalWrites: true });
60
+ return response.asReadonly();
61
+ });
48
62
  }
63
+ injectNgtLoader['preload'] = (loaderConstructorFactory, inputs, extensions) => {
64
+ Promise.all(load(loaderConstructorFactory, inputs, { extensions })());
65
+ };
49
66
  injectNgtLoader['destroy'] = () => {
50
67
  cached.clear();
51
68
  };
52
- injectNgtLoader['preLoad'] = (loaderConstructorFactory, inputs, extensions) => {
53
- load(loaderConstructorFactory, inputs, extensions).pipe(take(1)).subscribe();
54
- };
55
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"loader.js","sourceRoot":"","sources":["../../../../../libs/angular-three/src/lib/loader.ts"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, inject } from '@angular/core';\nimport {\n    ReplaySubject,\n    catchError,\n    forkJoin,\n    from,\n    isObservable,\n    map,\n    of,\n    retry,\n    share,\n    switchMap,\n    take,\n    tap,\n    type Observable,\n} from 'rxjs';\nimport type { GLTF } from 'three/examples/jsm/loaders/GLTFLoader';\nimport type {\n    NgtAnyRecord,\n    NgtBranchingReturn,\n    NgtLoaderExtensions,\n    NgtLoaderProto,\n    NgtLoaderReturnType,\n    NgtObjectMap,\n} from './types';\nimport { makeObjectGraph } from './utils/make';\nimport { safeDetectChanges } from './utils/safe-detect-changes';\n\nexport type NgtLoaderResults<\n    TInput extends string | string[] | Record<string, string>,\n    TReturn\n> = TInput extends string[] ? TReturn[] : TInput extends object ? { [key in keyof TInput]: TReturn } : TReturn;\n\nconst cached = new Map<string, Observable<any>>();\n\nfunction load<\n    TData,\n    TUrl extends string | string[] | Record<string, string>,\n    TLoaderConstructor extends NgtLoaderProto<TData>\n>(\n    loaderConstructorFactory: (inputs: TUrl) => TLoaderConstructor,\n    input: TUrl | Observable<TUrl>,\n    extensions?: NgtLoaderExtensions<TLoaderConstructor>,\n    onProgress?: (event: ProgressEvent) => void\n) {\n    const urls$ = isObservable(input) ? input : of(input);\n    return urls$.pipe(\n        map((inputs) => {\n            const loaderConstructor = loaderConstructorFactory(inputs);\n            const loader = new loaderConstructor();\n            if (extensions) extensions(loader);\n            const urls = Array.isArray(inputs) ? inputs : typeof inputs === 'string' ? [inputs] : Object.values(inputs);\n            return [\n                urls.map((url) => {\n                    if (!cached.has(url)) {\n                        cached.set(\n                            url,\n                            from(loader.loadAsync(url, onProgress)).pipe(\n                                tap((data) => {\n                                    if ((data as NgtAnyRecord)['scene'])\n                                        Object.assign(\n                                            data as NgtAnyRecord,\n                                            makeObjectGraph((data as NgtAnyRecord)['scene'])\n                                        );\n                                }),\n                                retry(2),\n                                catchError((err) => {\n                                    console.error(`[NGT] Error loading ${url}: ${err.message}`);\n                                    return of([]);\n                                }),\n                                share({ connector: () => new ReplaySubject(1) })\n                            )\n                        );\n                    }\n                    return cached.get(url);\n                }),\n                inputs,\n            ] as [Array<Observable<any>>, TUrl | TUrl[]];\n        })\n    );\n}\n\nexport function injectNgtLoader<\n    TData,\n    TUrl extends string | string[] | Record<string, string>,\n    TLoaderConstructor extends NgtLoaderProto<TData>,\n    TReturn = NgtLoaderReturnType<TData, TLoaderConstructor>\n>(\n    loaderConstructorFactory: (inputs: TUrl) => TLoaderConstructor,\n    input: TUrl | Observable<TUrl>,\n    extensions?: NgtLoaderExtensions<TLoaderConstructor>,\n    onProgress?: (event: ProgressEvent) => void\n): Observable<NgtLoaderResults<TUrl, NgtBranchingReturn<TReturn, GLTF, GLTF & NgtObjectMap>>> {\n    const cdr = inject(ChangeDetectorRef);\n\n    return load(loaderConstructorFactory, input, extensions, onProgress).pipe(\n        switchMap(([observables$, inputs]) => {\n            return forkJoin(observables$).pipe(\n                map((results) => {\n                    if (Array.isArray(inputs)) return results;\n                    if (typeof inputs === 'string') return results[0];\n                    const keys = Object.keys(inputs);\n                    return keys.reduce((result, key) => {\n                        result[key as keyof typeof result] = results[keys.indexOf(key)];\n                        return result;\n                    }, {} as { [key in keyof TUrl]: NgtBranchingReturn<TReturn, GLTF, GLTF & NgtObjectMap> });\n                }),\n                tap(() => {\n                    requestAnimationFrame(() => void safeDetectChanges(cdr));\n                })\n            );\n        })\n    );\n}\n\ninjectNgtLoader['destroy'] = () => {\n    cached.clear();\n};\n\ninjectNgtLoader['preLoad'] = <\n    TData,\n    TUrl extends string | string[] | Record<string, string>,\n    TLoaderConstructor extends NgtLoaderProto<TData>\n>(\n    loaderConstructorFactory: (inputs: TUrl) => TLoaderConstructor,\n    inputs: TUrl | Observable<TUrl>,\n    extensions?: NgtLoaderExtensions<TLoaderConstructor>\n) => {\n    load(loaderConstructorFactory, inputs, extensions).pipe(take(1)).subscribe();\n};\n"]}
69
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"loader.js","sourceRoot":"","sources":["../../../../../libs/angular-three/src/lib/loader.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectorRef,\n    Injector,\n    assertInInjectionContext,\n    computed,\n    effect,\n    inject,\n    runInInjectionContext,\n    signal,\n    untracked,\n    type Signal,\n} from '@angular/core';\nimport type { GLTF } from 'three/examples/jsm/loaders/GLTFLoader';\nimport type {\n    NgtAnyRecord,\n    NgtBranchingReturn,\n    NgtLoaderExtensions,\n    NgtLoaderProto,\n    NgtLoaderReturnType,\n    NgtObjectMap,\n} from './types';\nimport { makeObjectGraph } from './utils/make';\nimport { safeDetectChanges } from './utils/safe-detect-changes';\n\nexport type NgtLoaderResults<\n    TInput extends string | string[] | Record<string, string>,\n    TReturn\n> = TInput extends string[] ? TReturn[] : TInput extends object ? { [key in keyof TInput]: TReturn } : TReturn;\n\nconst cached = new Map();\n\nfunction load<\n    TData,\n    TUrl extends string | string[] | Record<string, string>,\n    TLoaderConstructor extends NgtLoaderProto<TData>\n>(\n    loaderConstructorFactory: (inputs: string[]) => TLoaderConstructor,\n    inputs: () => TUrl,\n    {\n        extensions,\n        onProgress,\n    }: {\n        extensions?: NgtLoaderExtensions<TLoaderConstructor>;\n        onProgress?: (event: ProgressEvent) => void;\n    } = {}\n) {\n    const computedUrls: Signal<string[]> = computed(() => {\n        const input = inputs();\n        if (Array.isArray(input)) return input;\n        if (typeof input === 'string') return [input];\n        return Object.values(input);\n    });\n\n    return () => {\n        const urls = computedUrls();\n        const loaderConstructor = loaderConstructorFactory(urls);\n        const loader = new loaderConstructor();\n        if (extensions) extensions(loader);\n\n        return urls.map(\n            (url) =>\n                new Promise((resolve, reject) => {\n                    if (cached.has(url)) {\n                        resolve(cached.get(url));\n                    } else {\n                        loader.load(\n                            url,\n                            (data) => {\n                                if ('scene' in (data as NgtAnyRecord))\n                                    Object.assign(\n                                        data as NgtAnyRecord,\n                                        makeObjectGraph((data as NgtAnyRecord)['scene'])\n                                    );\n                                cached.set(url, data);\n                                resolve(data);\n                            },\n                            onProgress,\n                            (error) => reject(new Error(`[NGT] Could not load ${url}: ${error}`))\n                        );\n                    }\n                })\n        );\n    };\n}\n\nexport function injectNgtLoader<\n    TData,\n    TUrl extends string | string[] | Record<string, string>,\n    TLoaderConstructor extends NgtLoaderProto<TData>,\n    TReturn = NgtLoaderReturnType<TData, TLoaderConstructor>\n>(\n    loaderConstructorFactory: (inputs: string[]) => TLoaderConstructor,\n    inputs: () => TUrl,\n    {\n        extensions,\n        onProgress,\n        injector = inject(Injector, { optional: true }),\n    }: {\n        extensions?: NgtLoaderExtensions<TLoaderConstructor>;\n        onProgress?: (event: ProgressEvent) => void;\n        injector?: Injector | null;\n    } = {}\n): Signal<NgtLoaderResults<TUrl, NgtBranchingReturn<TReturn, GLTF, GLTF & NgtObjectMap>>> {\n    !injector && assertInInjectionContext(injectNgtLoader);\n\n    return runInInjectionContext(injector!, () => {\n        const cdr = inject(ChangeDetectorRef);\n        const response = signal<NgtLoaderResults<TUrl, NgtBranchingReturn<TReturn, GLTF, GLTF & NgtObjectMap>>>(null!);\n        const effector = load(loaderConstructorFactory, inputs, { extensions, onProgress });\n\n        effect(\n            () => {\n                const originalUrls = untracked(inputs);\n                Promise.all(effector())\n                    .then((results) => {\n                        if (Array.isArray(originalUrls)) return results;\n                        if (typeof originalUrls === 'string') return results[0];\n                        const keys = Object.keys(originalUrls);\n                        return keys.reduce((result, key) => {\n                            (result as NgtAnyRecord)[key] = results[keys.indexOf(key)];\n                            return result;\n                        }, {} as { [key in keyof TUrl]: NgtBranchingReturn<TReturn, GLTF, GLTF & NgtObjectMap> });\n                    })\n                    .then((value) => {\n                        response.set(\n                            value as NgtLoaderResults<TUrl, NgtBranchingReturn<TReturn, GLTF, GLTF & NgtObjectMap>>\n                        );\n                        safeDetectChanges(cdr);\n                    });\n            },\n            { injector: injector!, allowSignalWrites: true }\n        );\n\n        return response.asReadonly();\n    });\n}\n\ninjectNgtLoader['preload'] = <\n    TData,\n    TUrl extends string | string[] | Record<string, string>,\n    TLoaderConstructor extends NgtLoaderProto<TData>\n>(\n    loaderConstructorFactory: (inputs: string[]) => TLoaderConstructor,\n    inputs: Signal<TUrl>,\n    extensions?: NgtLoaderExtensions<TLoaderConstructor>\n) => {\n    Promise.all(load(loaderConstructorFactory, inputs, { extensions })());\n};\n\ninjectNgtLoader['destroy'] = () => {\n    cached.clear();\n};\n"]}
@@ -49,15 +49,14 @@ function render(timestamp, store, frame) {
49
49
  state.clock.elapsedTime = timestamp;
50
50
  }
51
51
  // Call subscribers (useFrame)
52
- // subscribers = state.internal.subscribers;
53
- for (let i = 0; i < state.internal.subscribers.length; i++) {
54
- const subscriber = state.internal.subscribers[i];
55
- subscriber.callback({ ...state, delta, frame });
52
+ const subscribers = state.internal.subscribers;
53
+ for (let i = 0; i < subscribers.length; i++) {
54
+ const subscription = subscribers[i];
55
+ subscription.callback({ ...subscription.store.get(), delta, frame });
56
56
  }
57
57
  // Render content
58
- if (!state.internal.priority && state.gl.render) {
58
+ if (!state.internal.priority && state.gl.render)
59
59
  state.gl.render(state.scene, state.camera);
60
- }
61
60
  // Decrease frame count
62
61
  state.internal.frames = Math.max(0, state.internal.frames - 1);
63
62
  return state.frameloop === 'always' ? 1 : state.internal.frames;
@@ -114,7 +113,6 @@ export function createLoop(roots) {
114
113
  if (!state)
115
114
  for (const root of roots.values())
116
115
  render(timestamp, root);
117
- // safe to assume store is available here
118
116
  else
119
117
  render(timestamp, store, frame);
120
118
  if (runGlobalEffects)
@@ -134,4 +132,4 @@ export function createLoop(roots) {
134
132
  advance,
135
133
  };
136
134
  }
137
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"loop.js","sourceRoot":"","sources":["../../../../../libs/angular-three/src/lib/loop.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgtRxStore } from './stores/rx-store';\nimport type { NgtState } from './types';\n\nexport type NgtGlobalRenderCallback = (timestamp: number) => void;\ntype NgtSubItem = { callback: NgtGlobalRenderCallback };\n\nfunction createSubs(callback: NgtGlobalRenderCallback, subs: Set<NgtSubItem>): () => void {\n    const sub = { callback };\n    subs.add(sub);\n    return () => void subs.delete(sub);\n}\n\nconst globalEffects: Set<NgtSubItem> = new Set();\nconst globalAfterEffects: Set<NgtSubItem> = new Set();\nconst globalTailEffects: Set<NgtSubItem> = new Set();\n\n/**\n * Adds a global render callback which is called each frame.\n * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#addEffect\n */\nexport const addEffect = (callback: NgtGlobalRenderCallback) => createSubs(callback, globalEffects);\n\n/**\n * Adds a global after-render callback which is called each frame.\n * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#addAfterEffect\n */\nexport const addAfterEffect = (callback: NgtGlobalRenderCallback) => createSubs(callback, globalAfterEffects);\n\n/**\n * Adds a global callback which is called when rendering stops.\n * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#addTail\n */\nexport const addTail = (callback: NgtGlobalRenderCallback) => createSubs(callback, globalTailEffects);\n\nfunction run(effects: Set<NgtSubItem>, timestamp: number) {\n    if (!effects.size) return;\n    for (const { callback } of effects.values()) {\n        callback(timestamp);\n    }\n}\n\nexport type GlobalEffectType = 'before' | 'after' | 'tail';\n\nexport function flushGlobalEffects(type: GlobalEffectType, timestamp: number): void {\n    switch (type) {\n        case 'before':\n            return run(globalEffects, timestamp);\n        case 'after':\n            return run(globalAfterEffects, timestamp);\n        case 'tail':\n            return run(globalTailEffects, timestamp);\n    }\n}\n\nfunction render(timestamp: number, store: NgtRxStore<NgtState>, frame?: XRFrame) {\n    const state = store.get();\n    // Run local effects\n    let delta = state.clock.getDelta();\n    // In frameloop='never' mode, clock times are updated using the provided timestamp\n    if (state.frameloop === 'never' && typeof timestamp === 'number') {\n        delta = timestamp - state.clock.elapsedTime;\n        state.clock.oldTime = state.clock.elapsedTime;\n        state.clock.elapsedTime = timestamp;\n    }\n    // Call subscribers (useFrame)\n    // subscribers = state.internal.subscribers;\n    for (let i = 0; i < state.internal.subscribers.length; i++) {\n        const subscriber = state.internal.subscribers[i];\n        subscriber.callback({ ...state, delta, frame });\n    }\n    // Render content\n    if (!state.internal.priority && state.gl.render) {\n        state.gl.render(state.scene, state.camera);\n    }\n    // Decrease frame count\n    state.internal.frames = Math.max(0, state.internal.frames - 1);\n    return state.frameloop === 'always' ? 1 : state.internal.frames;\n}\n\nexport function createLoop<TCanvas>(roots: Map<TCanvas, NgtRxStore<NgtState>>) {\n    let running = false;\n    let repeat: number;\n    let frame: number;\n\n    function loop(timestamp: number): void {\n        frame = requestAnimationFrame(loop);\n        running = true;\n        repeat = 0;\n\n        // Run effects\n        flushGlobalEffects('before', timestamp);\n\n        // Render all roots\n        for (const root of roots.values()) {\n            const state = root.get();\n            // If the frameloop is invalidated, do not run another frame\n            if (\n                state.internal.active &&\n                (state.frameloop === 'always' || state.internal.frames > 0) &&\n                !state.gl.xr?.isPresenting\n            ) {\n                repeat += render(timestamp, root);\n            }\n        }\n\n        // Run after-effects\n        flushGlobalEffects('after', timestamp);\n\n        // Stop the loop if nothing invalidates it\n        if (repeat === 0) {\n            // Tail call effects, they are called when rendering stops\n            flushGlobalEffects('tail', timestamp);\n\n            // Flag end of operation\n            running = false;\n            return cancelAnimationFrame(frame);\n        }\n    }\n\n    function invalidate(store?: NgtRxStore<NgtState>, frames = 1): void {\n        const state = store?.get();\n        if (!state) return roots.forEach((root) => invalidate(root, frames));\n        if (state.gl.xr?.isPresenting || !state.internal.active || state.frameloop === 'never') return;\n        // Increase frames, do not go higher than 60\n        state.internal.frames = Math.min(60, state.internal.frames + frames);\n        // If the render-loop isn't active, start it\n        if (!running) {\n            running = true;\n            requestAnimationFrame(loop);\n        }\n    }\n\n    function advance(\n        timestamp: number,\n        runGlobalEffects: boolean = true,\n        store?: NgtRxStore<NgtState>,\n        frame?: XRFrame\n    ): void {\n        const state = store?.get();\n        if (runGlobalEffects) flushGlobalEffects('before', timestamp);\n        if (!state) for (const root of roots.values()) render(timestamp, root);\n        // safe to assume store is available here\n        else render(timestamp, store!, frame);\n        if (runGlobalEffects) flushGlobalEffects('after', timestamp);\n    }\n\n    return {\n        loop,\n        /**\n         * Invalidates the view, requesting a frame to be rendered. Will globally invalidate unless passed a root's state.\n         * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#invalidate\n         */\n        invalidate,\n        /**\n         * Advances the frameloop and runs render effects, useful for when manually rendering via `frameloop=\"never\"`.\n         * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#advance\n         */\n        advance,\n    };\n}\n"]}
135
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"loop.js","sourceRoot":"","sources":["../../../../../libs/angular-three/src/lib/loop.ts"],"names":[],"mappings":"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","sourcesContent":["import type { NgtSignalStore } from './stores/signal.store';\nimport type { NgtState } from './types';\n\nexport type NgtGlobalRenderCallback = (timestamp: number) => void;\ntype NgtSubItem = { callback: NgtGlobalRenderCallback };\n\nfunction createSubs(callback: NgtGlobalRenderCallback, subs: Set<NgtSubItem>): () => void {\n    const sub = { callback };\n    subs.add(sub);\n    return () => void subs.delete(sub);\n}\n\nconst globalEffects: Set<NgtSubItem> = new Set();\nconst globalAfterEffects: Set<NgtSubItem> = new Set();\nconst globalTailEffects: Set<NgtSubItem> = new Set();\n\n/**\n * Adds a global render callback which is called each frame.\n * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#addEffect\n */\nexport const addEffect = (callback: NgtGlobalRenderCallback) => createSubs(callback, globalEffects);\n\n/**\n * Adds a global after-render callback which is called each frame.\n * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#addAfterEffect\n */\nexport const addAfterEffect = (callback: NgtGlobalRenderCallback) => createSubs(callback, globalAfterEffects);\n\n/**\n * Adds a global callback which is called when rendering stops.\n * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#addTail\n */\nexport const addTail = (callback: NgtGlobalRenderCallback) => createSubs(callback, globalTailEffects);\n\nfunction run(effects: Set<NgtSubItem>, timestamp: number) {\n    if (!effects.size) return;\n    for (const { callback } of effects.values()) {\n        callback(timestamp);\n    }\n}\n\nexport type GlobalEffectType = 'before' | 'after' | 'tail';\n\nexport function flushGlobalEffects(type: GlobalEffectType, timestamp: number): void {\n    switch (type) {\n        case 'before':\n            return run(globalEffects, timestamp);\n        case 'after':\n            return run(globalAfterEffects, timestamp);\n        case 'tail':\n            return run(globalTailEffects, timestamp);\n    }\n}\n\nfunction render(timestamp: number, store: NgtSignalStore<NgtState>, frame?: XRFrame) {\n    const state = store.get();\n    // Run local effects\n    let delta = state.clock.getDelta();\n    // In frameloop='never' mode, clock times are updated using the provided timestamp\n    if (state.frameloop === 'never' && typeof timestamp === 'number') {\n        delta = timestamp - state.clock.elapsedTime;\n        state.clock.oldTime = state.clock.elapsedTime;\n        state.clock.elapsedTime = timestamp;\n    }\n    // Call subscribers (useFrame)\n    const subscribers = state.internal.subscribers;\n    for (let i = 0; i < subscribers.length; i++) {\n        const subscription = subscribers[i];\n        subscription.callback({ ...subscription.store.get(), delta, frame });\n    }\n    // Render content\n    if (!state.internal.priority && state.gl.render) state.gl.render(state.scene, state.camera);\n    // Decrease frame count\n    state.internal.frames = Math.max(0, state.internal.frames - 1);\n    return state.frameloop === 'always' ? 1 : state.internal.frames;\n}\n\nexport function createLoop<TCanvas>(roots: Map<TCanvas, NgtSignalStore<NgtState>>) {\n    let running = false;\n    let repeat: number;\n    let frame: number;\n\n    function loop(timestamp: number): void {\n        frame = requestAnimationFrame(loop);\n        running = true;\n        repeat = 0;\n\n        // Run effects\n        flushGlobalEffects('before', timestamp);\n\n        // Render all roots\n        for (const root of roots.values()) {\n            const state = root.get();\n            // If the frameloop is invalidated, do not run another frame\n            if (\n                state.internal.active &&\n                (state.frameloop === 'always' || state.internal.frames > 0) &&\n                !state.gl.xr?.isPresenting\n            ) {\n                repeat += render(timestamp, root);\n            }\n        }\n\n        // Run after-effects\n        flushGlobalEffects('after', timestamp);\n\n        // Stop the loop if nothing invalidates it\n        if (repeat === 0) {\n            // Tail call effects, they are called when rendering stops\n            flushGlobalEffects('tail', timestamp);\n\n            // Flag end of operation\n            running = false;\n            return cancelAnimationFrame(frame);\n        }\n    }\n\n    function invalidate(store?: NgtSignalStore<NgtState>, frames = 1): void {\n        const state = store?.get();\n        if (!state) return roots.forEach((root) => invalidate(root, frames));\n        if (state.gl.xr?.isPresenting || !state.internal.active || state.frameloop === 'never') return;\n        // Increase frames, do not go higher than 60\n        state.internal.frames = Math.min(60, state.internal.frames + frames);\n        // If the render-loop isn't active, start it\n        if (!running) {\n            running = true;\n            requestAnimationFrame(loop);\n        }\n    }\n\n    function advance(\n        timestamp: number,\n        runGlobalEffects: boolean = true,\n        store?: NgtSignalStore<NgtState>,\n        frame?: XRFrame\n    ): void {\n        const state = store?.get();\n        if (runGlobalEffects) flushGlobalEffects('before', timestamp);\n        if (!state) for (const root of roots.values()) render(timestamp, root);\n        else render(timestamp, store!, frame);\n        if (runGlobalEffects) flushGlobalEffects('after', timestamp);\n    }\n\n    return {\n        loop,\n        /**\n         * Invalidates the view, requesting a frame to be rendered. Will globally invalidate unless passed a root's state.\n         * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#invalidate\n         */\n        invalidate,\n        /**\n         * Advances the frameloop and runs render effects, useful for when manually rendering via `frameloop=\"never\"`.\n         * @see https://docs.pmnd.rs/react-three-fiber/api/additional-exports#advance\n         */\n        advance,\n    };\n}\n"]}
@@ -1,11 +1,13 @@
1
1
  import { NgIf } from '@angular/common';
2
- import { Component, ContentChild, Directive, EventEmitter, inject, Input, Output, SkipSelf, TemplateRef, ViewChild, ViewContainerRef, } from '@angular/core';
2
+ import { Component, ContentChild, DestroyRef, Directive, EventEmitter, Injector, Input, NgZone, Output, SkipSelf, TemplateRef, ViewChild, ViewContainerRef, effect, inject, } from '@angular/core';
3
3
  import * as THREE from 'three';
4
4
  import { injectNgtRef } from './di/ref';
5
- import { NgtRxStore } from './stores/rx-store';
5
+ import { SPECIAL_INTERNAL_ADD_COMMENT } from './renderer/utils';
6
+ import { NgtSignalStore } from './stores/signal.store';
6
7
  import { NgtStore } from './stores/store';
7
8
  import { getLocalState, prepare } from './utils/instance';
8
9
  import { is } from './utils/is';
10
+ import { safeDetectChanges } from './utils/safe-detect-changes';
9
11
  import { updateCamera } from './utils/update';
10
12
  import * as i0 from "@angular/core";
11
13
  const privateKeys = [
@@ -20,20 +22,23 @@ const privateKeys = [
20
22
  'advance',
21
23
  'size',
22
24
  'viewport',
23
- 'addInteraction',
24
- 'removeInteraction',
25
25
  ];
26
26
  class NgtPortalBeforeRender {
27
+ #portalStore;
28
+ #subscription;
27
29
  constructor() {
28
- this.portalStore = inject(NgtStore);
30
+ this.#portalStore = inject(NgtStore);
29
31
  this.renderPriority = 1;
30
32
  this.beforeRender = new EventEmitter();
33
+ inject(DestroyRef).onDestroy(() => {
34
+ this.#subscription?.();
35
+ });
31
36
  }
32
37
  ngOnInit() {
33
38
  let oldClear;
34
- this.subscription = this.portalStore.get('internal').subscribe(({ delta, frame }) => {
35
- this.beforeRender.emit({ ...this.portalStore.get(), delta, frame });
36
- const { gl, scene, camera } = this.portalStore.get();
39
+ this.#subscription = this.#portalStore.get('internal').subscribe(({ delta, frame }) => {
40
+ this.beforeRender.emit({ ...this.#portalStore.get(), delta, frame });
41
+ const { gl, scene, camera } = this.#portalStore.get();
37
42
  oldClear = gl.autoClear;
38
43
  if (this.renderPriority === 1) {
39
44
  // clear scene and render with default
@@ -46,10 +51,7 @@ class NgtPortalBeforeRender {
46
51
  gl.render(scene, camera);
47
52
  // restore
48
53
  gl.autoClear = oldClear;
49
- }, this.renderPriority, this.portalStore);
50
- }
51
- ngOnDestroy() {
52
- this.subscription?.();
54
+ }, this.renderPriority, this.#portalStore);
53
55
  }
54
56
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtPortalBeforeRender, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
55
57
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.0.0", type: NgtPortalBeforeRender, isStandalone: true, selector: "[ngtPortalBeforeRender]", inputs: { renderPriority: "renderPriority", parentScene: "parentScene", parentCamera: "parentCamera" }, outputs: { beforeRender: "beforeRender" }, ngImport: i0 }); }
@@ -58,21 +60,23 @@ export { NgtPortalBeforeRender };
58
60
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtPortalBeforeRender, decorators: [{
59
61
  type: Directive,
60
62
  args: [{ selector: '[ngtPortalBeforeRender]', standalone: true }]
61
- }], propDecorators: { renderPriority: [{
63
+ }], ctorParameters: function () { return []; }, propDecorators: { renderPriority: [{
62
64
  type: Input
63
65
  }], parentScene: [{
64
- type: Input
66
+ type: Input,
67
+ args: [{ required: true }]
65
68
  }], parentCamera: [{
66
- type: Input
69
+ type: Input,
70
+ args: [{ required: true }]
67
71
  }], beforeRender: [{
68
72
  type: Output
69
73
  }] } });
70
74
  class NgtPortalContent {
71
75
  constructor(vcr, parentVcr) {
72
76
  const commentNode = vcr.element.nativeElement;
73
- if (commentNode['__ngt_renderer_add_comment__']) {
74
- commentNode['__ngt_renderer_add_comment__'](parentVcr.element.nativeElement);
75
- delete commentNode['__ngt_renderer_add_comment__'];
77
+ if (commentNode[SPECIAL_INTERNAL_ADD_COMMENT]) {
78
+ commentNode[SPECIAL_INTERNAL_ADD_COMMENT](parentVcr.element.nativeElement);
79
+ delete commentNode[SPECIAL_INTERNAL_ADD_COMMENT];
76
80
  }
77
81
  }
78
82
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtPortalContent, deps: [{ token: i0.ViewContainerRef }, { token: i0.ViewContainerRef, skipSelf: true }], target: i0.ɵɵFactoryTarget.Directive }); }
@@ -85,32 +89,42 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImpor
85
89
  }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.ViewContainerRef, decorators: [{
86
90
  type: SkipSelf
87
91
  }] }]; } });
88
- class NgtPortal extends NgtRxStore {
89
- constructor() {
90
- super(...arguments);
91
- this.autoRender = true;
92
- this.autoRenderPriority = 1;
93
- this.beforeRender = new EventEmitter();
94
- this.parentStore = inject(NgtStore, { skipSelf: true });
95
- this.parentScene = this.parentStore.get('scene');
96
- this.parentCamera = this.parentStore.get('camera');
97
- this.portalStore = inject(NgtStore, { self: true });
98
- this.raycaster = new THREE.Raycaster();
99
- this.pointer = new THREE.Vector2();
100
- this.portalContentRendered = false;
101
- }
92
+ class NgtPortal extends NgtSignalStore {
102
93
  set container(container) {
103
94
  this.set({ container });
104
95
  }
105
96
  set state(state) {
106
97
  this.set({ state });
107
98
  }
108
- initialize() {
109
- super.initialize();
110
- this.set({ container: injectNgtRef(prepare(new THREE.Scene())) });
99
+ #parentStore;
100
+ #portalStore;
101
+ #injector;
102
+ #zone;
103
+ #raycaster;
104
+ #pointer;
105
+ #portalContentView;
106
+ constructor() {
107
+ super({ container: injectNgtRef(prepare(new THREE.Scene())) });
108
+ this.autoRender = true;
109
+ this.autoRenderPriority = 1;
110
+ this.beforeRender = new EventEmitter();
111
+ this.#parentStore = inject(NgtStore, { skipSelf: true });
112
+ this.parentScene = this.#parentStore.get('scene');
113
+ this.parentCamera = this.#parentStore.get('camera');
114
+ this.#portalStore = inject(NgtStore, { self: true });
115
+ this.#injector = inject(Injector);
116
+ this.#zone = inject(NgZone);
117
+ this.#raycaster = new THREE.Raycaster();
118
+ this.#pointer = new THREE.Vector2();
119
+ this.portalContentRendered = false;
120
+ inject(DestroyRef).onDestroy(() => {
121
+ if (this.#portalContentView && !this.#portalContentView.destroyed) {
122
+ this.#portalContentView.destroy();
123
+ }
124
+ });
111
125
  }
112
126
  ngOnInit() {
113
- const previousState = this.parentStore.get();
127
+ const previousState = this.#parentStore.get();
114
128
  const inputsState = this.get();
115
129
  if (!inputsState.state && this.autoRender) {
116
130
  inputsState.state = { events: { priority: this.autoRenderPriority + 1 } };
@@ -120,43 +134,42 @@ class NgtPortal extends NgtRxStore {
120
134
  const container = is.ref(containerState) ? containerState.nativeElement : containerState;
121
135
  const localState = getLocalState(container);
122
136
  if (!localState.store) {
123
- localState.store = this.portalStore;
137
+ localState.store = this.#portalStore;
124
138
  }
125
- this.portalStore.set({
139
+ this.#portalStore.set({
126
140
  ...previousState,
127
141
  scene: container,
128
- raycaster: this.raycaster,
129
- pointer: this.pointer,
130
- previousStore: this.parentStore,
142
+ raycaster: this.#raycaster,
143
+ pointer: this.#pointer,
144
+ previousStore: this.#parentStore,
131
145
  events: { ...previousState.events, ...(events || {}) },
132
146
  size: { ...previousState.size, ...(size || {}) },
133
147
  ...restInputsState,
134
- get: this.portalStore.get.bind(this.portalStore),
135
- set: this.portalStore.set.bind(this.portalStore),
136
- select: this.portalStore.select.bind(this.portalStore),
137
- setEvents: (events) => this.portalStore.set((state) => ({ ...state, events: { ...state.events, ...events } })),
148
+ get: this.#portalStore.get.bind(this.#portalStore),
149
+ set: this.#portalStore.set.bind(this.#portalStore),
150
+ setEvents: (events) => this.#portalStore.set((state) => ({ ...state, events: { ...state.events, ...events } })),
138
151
  });
139
- this.hold(this.parentStore.select(), (previous) => this.portalStore.set((state) => this.inject(previous, state)));
152
+ const parentState = this.#parentStore.select();
153
+ effect(() => {
154
+ const previous = parentState();
155
+ this.#zone.runOutsideAngular(() => {
156
+ this.#portalStore.set((state) => this.#inject(previous, state));
157
+ });
158
+ }, { injector: this.#injector, allowSignalWrites: true });
140
159
  requestAnimationFrame(() => {
141
- this.portalStore.set((injectState) => this.inject(this.parentStore.get(), injectState));
160
+ this.#portalStore.set((injectState) => this.#inject(this.#parentStore.get(), injectState));
142
161
  });
143
- this.portalContentView = this.portalContentAnchor.createEmbeddedView(this.portalContentTemplate);
144
- this.portalContentView.detectChanges();
162
+ this.#portalContentView = this.portalContentAnchor.createEmbeddedView(this.portalContentTemplate);
163
+ safeDetectChanges(this.#portalContentView);
145
164
  this.portalContentRendered = true;
146
165
  }
147
166
  onBeforeRender(portal) {
148
167
  this.beforeRender.emit({
149
- root: { ...this.parentStore.get(), delta: portal.delta, frame: portal.frame },
168
+ root: { ...this.#parentStore.get(), delta: portal.delta, frame: portal.frame },
150
169
  portal,
151
170
  });
152
171
  }
153
- ngOnDestroy() {
154
- if (this.portalContentView && !this.portalContentView.destroyed) {
155
- this.portalContentView.destroy();
156
- }
157
- super.ngOnDestroy();
158
- }
159
- inject(rootState, injectState) {
172
+ #inject(rootState, injectState) {
160
173
  const intersect = { ...rootState };
161
174
  Object.keys(intersect).forEach((key) => {
162
175
  if (privateKeys.includes(key) ||
@@ -176,16 +189,16 @@ class NgtPortal extends NgtRxStore {
176
189
  return {
177
190
  ...intersect,
178
191
  scene: is.ref(inputs.container) ? inputs.container.nativeElement : inputs.container,
179
- raycaster: this.raycaster,
180
- pointer: this.pointer,
181
- previousStore: this.parentStore,
192
+ raycaster: this.#raycaster,
193
+ pointer: this.#pointer,
194
+ previousStore: this.#parentStore,
182
195
  events: { ...rootState.events, ...(injectState?.events || {}), ...events },
183
196
  size: { ...rootState.size, ...size },
184
197
  viewport: { ...rootState.viewport, ...(viewport || {}) },
185
198
  ...restInputsState,
186
199
  };
187
200
  }
188
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtPortal, deps: null, target: i0.ɵɵFactoryTarget.Component }); }
201
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgtPortal, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
189
202
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: NgtPortal, isStandalone: true, selector: "ngt-portal", inputs: { container: "container", state: "state", autoRender: "autoRender", autoRenderPriority: "autoRenderPriority" }, outputs: { beforeRender: "beforeRender" }, providers: [NgtStore], queries: [{ propertyName: "portalContentTemplate", first: true, predicate: NgtPortalContent, descendants: true, read: TemplateRef, static: true }], viewQueries: [{ propertyName: "portalContentAnchor", first: true, predicate: ["portalContentAnchor"], descendants: true, read: ViewContainerRef, static: true }], usesInheritance: true, ngImport: i0, template: `
190
203
  <ng-container #portalContentAnchor>
191
204
  <ng-container
@@ -220,7 +233,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImpor
220
233
  imports: [NgIf, NgtPortalBeforeRender],
221
234
  providers: [NgtStore],
222
235
  }]
223
- }], propDecorators: { container: [{
236
+ }], ctorParameters: function () { return []; }, propDecorators: { container: [{
224
237
  type: Input
225
238
  }], state: [{
226
239
  type: Input
@@ -237,4 +250,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImpor
237
250
  type: ViewChild,
238
251
  args: ['portalContentAnchor', { read: ViewContainerRef, static: true }]
239
252
  }] } });
240
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"portal.js","sourceRoot":"","sources":["../../../../../libs/angular-three/src/lib/portal.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,iBAAiB,CAAC;AACvC,OAAO,EACH,SAAS,EACT,YAAY,EACZ,SAAS,EAGT,YAAY,EACZ,MAAM,EACN,KAAK,EAGL,MAAM,EACN,QAAQ,EACR,WAAW,EACX,SAAS,EACT,gBAAgB,GACnB,MAAM,eAAe,CAAC;AACvB,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,EAAE,YAAY,EAAE,MAAM,UAAU,CAAC;AACxC,OAAO,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAC/C,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE1C,OAAO,EAAE,aAAa,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAC1D,OAAO,EAAE,EAAE,EAAE,MAAM,YAAY,CAAC;AAChC,OAAO,EAAE,YAAY,EAAE,MAAM,gBAAgB,CAAC;;AAE9C,MAAM,WAAW,GAAG;IAChB,KAAK;IACL,KAAK;IACL,QAAQ;IACR,SAAS;IACT,QAAQ;IACR,cAAc;IACd,QAAQ;IACR,YAAY;IACZ,SAAS;IACT,MAAM;IACN,UAAU;IACV,gBAAgB;IAChB,mBAAmB;CACb,CAAC;AAcX,MACa,qBAAqB;IADlC;QAEqB,gBAAW,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;QAEvC,mBAAc,GAAG,CAAC,CAAC;QAIlB,iBAAY,GAAG,IAAI,YAAY,EAAkB,CAAC;KA+B/D;IA3BG,QAAQ;QACJ,IAAI,QAAiB,CAAC;QACtB,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,SAAS,CAC1D,CAAC,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE,EAAE;YACjB,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC;YACpE,MAAM,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,CAAC;YACrD,QAAQ,GAAG,EAAE,CAAC,SAAS,CAAC;YACxB,IAAI,IAAI,CAAC,cAAc,KAAK,CAAC,EAAE;gBAC3B,sCAAsC;gBACtC,EAAE,CAAC,SAAS,GAAG,IAAI,CAAC;gBACpB,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC;aAClD;YACD,mBAAmB;YACnB,EAAE,CAAC,SAAS,GAAG,KAAK,CAAC;YACrB,EAAE,CAAC,UAAU,EAAE,CAAC;YAChB,EAAE,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YACzB,UAAU;YACV,EAAE,CAAC,SAAS,GAAG,QAAQ,CAAC;QAC5B,CAAC,EACD,IAAI,CAAC,cAAc,EACnB,IAAI,CAAC,WAAW,CACnB,CAAC;IACN,CAAC;IAED,WAAW;QACP,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;IAC1B,CAAC;8GArCQ,qBAAqB;kGAArB,qBAAqB;;SAArB,qBAAqB;2FAArB,qBAAqB;kBADjC,SAAS;mBAAC,EAAE,QAAQ,EAAE,yBAAyB,EAAE,UAAU,EAAE,IAAI,EAAE;8BAIvD,cAAc;sBAAtB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBAEI,YAAY;sBAArB,MAAM;;AAiCX,MACa,gBAAgB;IACzB,YAAY,GAAqB,EAAc,SAA2B;QACtE,MAAM,WAAW,GAAG,GAAG,CAAC,OAAO,CAAC,aAAa,CAAC;QAC9C,IAAI,WAAW,CAAC,8BAA8B,CAAC,EAAE;YAC7C,WAAW,CAAC,8BAA8B,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;YAC7E,OAAO,WAAW,CAAC,8BAA8B,CAAC,CAAC;SACtD;IACL,CAAC;8GAPQ,gBAAgB;kGAAhB,gBAAgB;;SAAhB,gBAAgB;2FAAhB,gBAAgB;kBAD5B,SAAS;mBAAC,EAAE,QAAQ,EAAE,+BAA+B,EAAE,UAAU,EAAE,IAAI,EAAE;;0BAElC,QAAQ;;AAShD,MAkBa,SAAU,SAAQ,UAA2B;IAlB1D;;QA2Ba,eAAU,GAAG,IAAI,CAAC;QAClB,uBAAkB,GAAG,CAAC,CAAC;QAEtB,iBAAY,GAAG,IAAI,YAAY,EAAoD,CAAC;QAQ7E,gBAAW,GAAG,MAAM,CAAC,QAAQ,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;QAC3D,gBAAW,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAC5C,iBAAY,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAEtC,gBAAW,GAAG,MAAM,CAAC,QAAQ,EAAE,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;QAE/C,cAAS,GAAG,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;QAClC,YAAO,GAAG,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;QAE/C,0BAAqB,GAAG,KAAK,CAAC;KAsGjC;IAlIG,IAAa,SAAS,CAAC,SAAuC;QAC1D,IAAI,CAAC,GAAG,CAAC,EAAE,SAAS,EAAE,CAAC,CAAC;IAC5B,CAAC;IAED,IAAa,KAAK,CAAC,KAA+B;QAC9C,IAAI,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;IACxB,CAAC;IAyBQ,UAAU;QACf,KAAK,CAAC,UAAU,EAAE,CAAC;QACnB,IAAI,CAAC,GAAG,CAAC,EAAE,SAAS,EAAE,YAAY,CAAc,OAAO,CAAC,IAAI,KAAK,CAAC,KAAK,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC;IACnF,CAAC;IAED,QAAQ;QACJ,MAAM,aAAa,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,CAAC;QAC7C,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAE/B,IAAI,CAAC,WAAW,CAAC,KAAK,IAAI,IAAI,CAAC,UAAU,EAAE;YACvC,WAAW,CAAC,KAAK,GAAG,EAAE,MAAM,EAAE,EAAE,QAAQ,EAAE,IAAI,CAAC,kBAAkB,GAAG,CAAC,EAAE,EAAE,CAAC;SAC7E;QAED,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,eAAe,EAAE,GAAG,WAAW,CAAC,KAAK,IAAI,EAAE,CAAC;QAErE,MAAM,cAAc,GAAG,WAAW,CAAC,SAAS,CAAC;QAC7C,MAAM,SAAS,GAAG,EAAE,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,cAAc,CAAC,aAAa,CAAC,CAAC,CAAC,cAAc,CAAC;QAEzF,MAAM,UAAU,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC;QAC5C,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE;YACnB,UAAU,CAAC,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC;SACvC;QAED,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC;YACjB,GAAG,aAAa;YAChB,KAAK,EAAE,SAAwB;YAC/B,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,aAAa,EAAE,IAAI,CAAC,WAAW;YAC/B,MAAM,EAAE,EAAE,GAAG,aAAa,CAAC,MAAM,EAAE,GAAG,CAAC,MAAM,IAAI,EAAE,CAAC,EAAE;YACtD,IAAI,EAAE,EAAE,GAAG,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,IAAI,IAAI,EAAE,CAAC,EAAE;YAChD,GAAG,eAAe;YAClB,GAAG,EAAE,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;YAChD,GAAG,EAAE,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;YAChD,MAAM,EAAE,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;YACtD,SAAS,EAAE,CAAC,MAAM,EAAE,EAAE,CAClB,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,KAAK,EAAE,MAAM,EAAE,EAAE,GAAG,KAAK,CAAC,MAAM,EAAE,GAAG,MAAM,EAAE,EAAE,CAAC,CAAC;SAC9F,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,EAAE,EAAE,CAAC,QAAQ,EAAE,EAAE,CAC9C,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC,CAChE,CAAC;QAEF,qBAAqB,CAAC,GAAG,EAAE;YACvB,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,WAAW,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,WAAW,CAAC,CAAC,CAAC;QAC5F,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;QACjG,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;QACvC,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC;IACtC,CAAC;IAED,cAAc,CAAC,MAAsB;QACjC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;YACnB,IAAI,EAAE,EAAE,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE;YAC7E,MAAM;SACT,CAAC,CAAC;IACP,CAAC;IAEQ,WAAW;QAChB,IAAI,IAAI,CAAC,iBAAiB,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,SAAS,EAAE;YAC7D,IAAI,CAAC,iBAAiB,CAAC,OAAO,EAAE,CAAC;SACpC;QACD,KAAK,CAAC,WAAW,EAAE,CAAC;IACxB,CAAC;IAEO,MAAM,CAAC,SAAmB,EAAE,WAAqB;QACrD,MAAM,SAAS,GAAsB,EAAE,GAAG,SAAS,EAAE,CAAC;QAEtD,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;YACnC,IACI,WAAW,CAAC,QAAQ,CAAC,GAAkB,CAAC;gBACxC,SAAS,CAAC,GAAqB,CAAC,KAAK,WAAW,CAAC,GAAqB,CAAC,EACzE;gBACE,OAAO,SAAS,CAAC,GAAqB,CAAC,CAAC;aAC3C;QACL,CAAC,CAAC,CAAC;QAEH,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC1B,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,eAAe,EAAE,GAAG,MAAM,CAAC,KAAK,IAAI,EAAE,CAAC;QAEhE,IAAI,QAAQ,GAAG,SAAS,CAAC;QACzB,IAAI,WAAW,IAAI,IAAI,EAAE;YACrB,MAAM,MAAM,GAAG,WAAW,CAAC,MAAM,CAAC;YAClC,QAAQ,GAAG,SAAS,CAAC,QAAQ,CAAC,kBAAkB,CAAC,MAAM,EAAE,IAAI,KAAK,CAAC,OAAO,EAAE,EAAE,IAAI,CAAC,CAAC;YACpF,IAAI,MAAM,KAAK,SAAS,CAAC,MAAM;gBAAE,YAAY,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;SAC/D;QAED,OAAO;YACH,GAAG,SAAS;YACZ,KAAK,EAAE,EAAE,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS;YACnF,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,aAAa,EAAE,IAAI,CAAC,WAAW;YAC/B,MAAM,EAAE,EAAE,GAAG,SAAS,CAAC,MAAM,EAAE,GAAG,CAAC,WAAW,EAAE,MAAM,IAAI,EAAE,CAAC,EAAE,GAAG,MAAM,EAAE;YAC1E,IAAI,EAAE,EAAE,GAAG,SAAS,CAAC,IAAI,EAAE,GAAG,IAAI,EAAE;YACpC,QAAQ,EAAE,EAAE,GAAG,SAAS,CAAC,QAAQ,EAAE,GAAG,CAAC,QAAQ,IAAI,EAAE,CAAC,EAAE;YACxD,GAAG,eAAe;SACT,CAAC;IAClB,CAAC;8GAlIQ,SAAS;kGAAT,SAAS,4NAFP,CAAC,QAAQ,CAAC,6EAgBP,gBAAgB,2BAAU,WAAW,kJAGT,gBAAgB,kEAhChD;;;;;;;;;;;KAWT,4DACS,IAAI,6FAlEL,qBAAqB;;SAqErB,SAAS;2FAAT,SAAS;kBAlBrB,SAAS;mBAAC;oBACP,QAAQ,EAAE,YAAY;oBACtB,UAAU,EAAE,IAAI;oBAChB,QAAQ,EAAE;;;;;;;;;;;KAWT;oBACD,OAAO,EAAE,CAAC,IAAI,EAAE,qBAAqB,CAAC;oBACtC,SAAS,EAAE,CAAC,QAAQ,CAAC;iBACxB;8BAEgB,SAAS;sBAArB,KAAK;gBAIO,KAAK;sBAAjB,KAAK;gBAIG,UAAU;sBAAlB,KAAK;gBACG,kBAAkB;sBAA1B,KAAK;gBAEI,YAAY;sBAArB,MAAM;gBAGE,qBAAqB;sBAD7B,YAAY;uBAAC,gBAAgB,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE;gBAI1D,mBAAmB;sBAD3B,SAAS;uBAAC,qBAAqB,EAAE,EAAE,IAAI,EAAE,gBAAgB,EAAE,MAAM,EAAE,IAAI,EAAE","sourcesContent":["import { NgIf } from '@angular/common';\nimport {\n    Component,\n    ContentChild,\n    Directive,\n    ElementRef,\n    EmbeddedViewRef,\n    EventEmitter,\n    inject,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    SkipSelf,\n    TemplateRef,\n    ViewChild,\n    ViewContainerRef,\n} from '@angular/core';\nimport * as THREE from 'three';\nimport { injectNgtRef } from './di/ref';\nimport { NgtRxStore } from './stores/rx-store';\nimport { NgtStore } from './stores/store';\nimport { NgtEventManager, NgtRenderState, NgtSize, NgtState } from './types';\nimport { getLocalState, prepare } from './utils/instance';\nimport { is } from './utils/is';\nimport { updateCamera } from './utils/update';\n\nconst privateKeys = [\n    'get',\n    'set',\n    'select',\n    'setSize',\n    'setDpr',\n    'setFrameloop',\n    'events',\n    'invalidate',\n    'advance',\n    'size',\n    'viewport',\n    'addInteraction',\n    'removeInteraction',\n] as const;\ntype PrivateKeys = (typeof privateKeys)[number];\n\nexport interface NgtPortalInputs {\n    container: ElementRef<THREE.Object3D> | THREE.Object3D;\n    camera: ElementRef<THREE.Camera> | THREE.Camera;\n    state: Partial<\n        Omit<NgtState, PrivateKeys> & {\n            events: Partial<Pick<NgtEventManager<any>, 'enabled' | 'priority' | 'compute' | 'connected'>>;\n            size: NgtSize;\n        }\n    >;\n}\n\n@Directive({ selector: '[ngtPortalBeforeRender]', standalone: true })\nexport class NgtPortalBeforeRender implements OnInit, OnDestroy {\n    private readonly portalStore = inject(NgtStore);\n\n    @Input() renderPriority = 1;\n    @Input() parentScene!: THREE.Scene;\n    @Input() parentCamera!: THREE.Camera;\n\n    @Output() beforeRender = new EventEmitter<NgtRenderState>();\n\n    private subscription?: () => void;\n\n    ngOnInit() {\n        let oldClear: boolean;\n        this.subscription = this.portalStore.get('internal').subscribe(\n            ({ delta, frame }) => {\n                this.beforeRender.emit({ ...this.portalStore.get(), delta, frame });\n                const { gl, scene, camera } = this.portalStore.get();\n                oldClear = gl.autoClear;\n                if (this.renderPriority === 1) {\n                    // clear scene and render with default\n                    gl.autoClear = true;\n                    gl.render(this.parentScene, this.parentCamera);\n                }\n                // disable cleaning\n                gl.autoClear = false;\n                gl.clearDepth();\n                gl.render(scene, camera);\n                // restore\n                gl.autoClear = oldClear;\n            },\n            this.renderPriority,\n            this.portalStore\n        );\n    }\n\n    ngOnDestroy() {\n        this.subscription?.();\n    }\n}\n\n@Directive({ selector: 'ng-template[ngtPortalContent]', standalone: true })\nexport class NgtPortalContent {\n    constructor(vcr: ViewContainerRef, @SkipSelf() parentVcr: ViewContainerRef) {\n        const commentNode = vcr.element.nativeElement;\n        if (commentNode['__ngt_renderer_add_comment__']) {\n            commentNode['__ngt_renderer_add_comment__'](parentVcr.element.nativeElement);\n            delete commentNode['__ngt_renderer_add_comment__'];\n        }\n    }\n}\n\n@Component({\n    selector: 'ngt-portal',\n    standalone: true,\n    template: `\n        <ng-container #portalContentAnchor>\n            <ng-container\n                *ngIf=\"autoRender && portalContentRendered\"\n                ngtPortalBeforeRender\n                [renderPriority]=\"autoRenderPriority\"\n                [parentScene]=\"parentScene\"\n                [parentCamera]=\"parentCamera\"\n                (beforeRender)=\"onBeforeRender($event)\"\n            />\n        </ng-container>\n    `,\n    imports: [NgIf, NgtPortalBeforeRender],\n    providers: [NgtStore],\n})\nexport class NgtPortal extends NgtRxStore<NgtPortalInputs> implements OnInit, OnDestroy {\n    @Input() set container(container: NgtPortalInputs['container']) {\n        this.set({ container });\n    }\n\n    @Input() set state(state: NgtPortalInputs['state']) {\n        this.set({ state });\n    }\n\n    @Input() autoRender = true;\n    @Input() autoRenderPriority = 1;\n\n    @Output() beforeRender = new EventEmitter<{ root: NgtRenderState; portal: NgtRenderState }>();\n\n    @ContentChild(NgtPortalContent, { read: TemplateRef, static: true })\n    readonly portalContentTemplate!: TemplateRef<unknown>;\n\n    @ViewChild('portalContentAnchor', { read: ViewContainerRef, static: true })\n    readonly portalContentAnchor!: ViewContainerRef;\n\n    private readonly parentStore = inject(NgtStore, { skipSelf: true });\n    readonly parentScene = this.parentStore.get('scene');\n    readonly parentCamera = this.parentStore.get('camera');\n\n    private readonly portalStore = inject(NgtStore, { self: true });\n\n    private readonly raycaster = new THREE.Raycaster();\n    private readonly pointer = new THREE.Vector2();\n\n    portalContentRendered = false;\n    private portalContentView?: EmbeddedViewRef<unknown>;\n\n    override initialize() {\n        super.initialize();\n        this.set({ container: injectNgtRef<THREE.Scene>(prepare(new THREE.Scene())) });\n    }\n\n    ngOnInit() {\n        const previousState = this.parentStore.get();\n        const inputsState = this.get();\n\n        if (!inputsState.state && this.autoRender) {\n            inputsState.state = { events: { priority: this.autoRenderPriority + 1 } };\n        }\n\n        const { events, size, ...restInputsState } = inputsState.state || {};\n\n        const containerState = inputsState.container;\n        const container = is.ref(containerState) ? containerState.nativeElement : containerState;\n\n        const localState = getLocalState(container);\n        if (!localState.store) {\n            localState.store = this.portalStore;\n        }\n\n        this.portalStore.set({\n            ...previousState,\n            scene: container as THREE.Scene,\n            raycaster: this.raycaster,\n            pointer: this.pointer,\n            previousStore: this.parentStore,\n            events: { ...previousState.events, ...(events || {}) },\n            size: { ...previousState.size, ...(size || {}) },\n            ...restInputsState,\n            get: this.portalStore.get.bind(this.portalStore),\n            set: this.portalStore.set.bind(this.portalStore),\n            select: this.portalStore.select.bind(this.portalStore),\n            setEvents: (events) =>\n                this.portalStore.set((state) => ({ ...state, events: { ...state.events, ...events } })),\n        });\n\n        this.hold(this.parentStore.select(), (previous) =>\n            this.portalStore.set((state) => this.inject(previous, state))\n        );\n\n        requestAnimationFrame(() => {\n            this.portalStore.set((injectState) => this.inject(this.parentStore.get(), injectState));\n        });\n        this.portalContentView = this.portalContentAnchor.createEmbeddedView(this.portalContentTemplate);\n        this.portalContentView.detectChanges();\n        this.portalContentRendered = true;\n    }\n\n    onBeforeRender(portal: NgtRenderState) {\n        this.beforeRender.emit({\n            root: { ...this.parentStore.get(), delta: portal.delta, frame: portal.frame },\n            portal,\n        });\n    }\n\n    override ngOnDestroy() {\n        if (this.portalContentView && !this.portalContentView.destroyed) {\n            this.portalContentView.destroy();\n        }\n        super.ngOnDestroy();\n    }\n\n    private inject(rootState: NgtState, injectState: NgtState) {\n        const intersect: Partial<NgtState> = { ...rootState };\n\n        Object.keys(intersect).forEach((key) => {\n            if (\n                privateKeys.includes(key as PrivateKeys) ||\n                rootState[key as keyof NgtState] !== injectState[key as keyof NgtState]\n            ) {\n                delete intersect[key as keyof NgtState];\n            }\n        });\n\n        const inputs = this.get();\n        const { size, events, ...restInputsState } = inputs.state || {};\n\n        let viewport = undefined;\n        if (injectState && size) {\n            const camera = injectState.camera;\n            viewport = rootState.viewport.getCurrentViewport(camera, new THREE.Vector3(), size);\n            if (camera !== rootState.camera) updateCamera(camera, size);\n        }\n\n        return {\n            ...intersect,\n            scene: is.ref(inputs.container) ? inputs.container.nativeElement : inputs.container,\n            raycaster: this.raycaster,\n            pointer: this.pointer,\n            previousStore: this.parentStore,\n            events: { ...rootState.events, ...(injectState?.events || {}), ...events },\n            size: { ...rootState.size, ...size },\n            viewport: { ...rootState.viewport, ...(viewport || {}) },\n            ...restInputsState,\n        } as NgtState;\n    }\n}\n"]}
253
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"portal.js","sourceRoot":"","sources":["../../../../../libs/angular-three/src/lib/portal.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgIf } from '@angular/common';\nimport {\n    Component,\n    ContentChild,\n    DestroyRef,\n    Directive,\n    EventEmitter,\n    Injector,\n    Input,\n    NgZone,\n    Output,\n    SkipSelf,\n    TemplateRef,\n    ViewChild,\n    ViewContainerRef,\n    effect,\n    inject,\n    type ElementRef,\n    type EmbeddedViewRef,\n    type OnInit,\n} from '@angular/core';\nimport * as THREE from 'three';\nimport { injectNgtRef } from './di/ref';\nimport { SPECIAL_INTERNAL_ADD_COMMENT } from './renderer/utils';\nimport { NgtSignalStore } from './stores/signal.store';\nimport { NgtStore } from './stores/store';\nimport type { NgtEventManager, NgtRenderState, NgtSize, NgtState } from './types';\nimport { getLocalState, prepare } from './utils/instance';\nimport { is } from './utils/is';\nimport { safeDetectChanges } from './utils/safe-detect-changes';\nimport { updateCamera } from './utils/update';\n\nconst privateKeys = [\n    'get',\n    'set',\n    'select',\n    'setSize',\n    'setDpr',\n    'setFrameloop',\n    'events',\n    'invalidate',\n    'advance',\n    'size',\n    'viewport',\n] as const;\ntype PrivateKeys = (typeof privateKeys)[number];\n\nexport interface NgtPortalInputs {\n    container: ElementRef<THREE.Object3D> | THREE.Object3D;\n    camera: ElementRef<THREE.Camera> | THREE.Camera;\n    state: Partial<\n        Omit<NgtState, PrivateKeys> & {\n            events: Partial<Pick<NgtEventManager<any>, 'enabled' | 'priority' | 'compute' | 'connected'>>;\n            size: NgtSize;\n        }\n    >;\n}\n\n@Directive({ selector: '[ngtPortalBeforeRender]', standalone: true })\nexport class NgtPortalBeforeRender implements OnInit {\n    readonly #portalStore = inject(NgtStore);\n\n    @Input() renderPriority = 1;\n    @Input({ required: true }) parentScene!: THREE.Scene;\n    @Input({ required: true }) parentCamera!: THREE.Camera;\n\n    @Output() beforeRender = new EventEmitter<NgtRenderState>();\n\n    #subscription?: () => void;\n\n    constructor() {\n        inject(DestroyRef).onDestroy(() => {\n            this.#subscription?.();\n        });\n    }\n\n    ngOnInit() {\n        let oldClear: boolean;\n        this.#subscription = this.#portalStore.get('internal').subscribe(\n            ({ delta, frame }) => {\n                this.beforeRender.emit({ ...this.#portalStore.get(), delta, frame });\n                const { gl, scene, camera } = this.#portalStore.get();\n                oldClear = gl.autoClear;\n                if (this.renderPriority === 1) {\n                    // clear scene and render with default\n                    gl.autoClear = true;\n                    gl.render(this.parentScene, this.parentCamera);\n                }\n                // disable cleaning\n                gl.autoClear = false;\n                gl.clearDepth();\n                gl.render(scene, camera);\n                // restore\n                gl.autoClear = oldClear;\n            },\n            this.renderPriority,\n            this.#portalStore\n        );\n    }\n}\n\n@Directive({ selector: 'ng-template[ngtPortalContent]', standalone: true })\nexport class NgtPortalContent {\n    constructor(vcr: ViewContainerRef, @SkipSelf() parentVcr: ViewContainerRef) {\n        const commentNode = vcr.element.nativeElement;\n        if (commentNode[SPECIAL_INTERNAL_ADD_COMMENT]) {\n            commentNode[SPECIAL_INTERNAL_ADD_COMMENT](parentVcr.element.nativeElement);\n            delete commentNode[SPECIAL_INTERNAL_ADD_COMMENT];\n        }\n    }\n}\n\n@Component({\n    selector: 'ngt-portal',\n    standalone: true,\n    template: `\n        <ng-container #portalContentAnchor>\n            <ng-container\n                *ngIf=\"autoRender && portalContentRendered\"\n                ngtPortalBeforeRender\n                [renderPriority]=\"autoRenderPriority\"\n                [parentScene]=\"parentScene\"\n                [parentCamera]=\"parentCamera\"\n                (beforeRender)=\"onBeforeRender($event)\"\n            />\n        </ng-container>\n    `,\n    imports: [NgIf, NgtPortalBeforeRender],\n    providers: [NgtStore],\n})\nexport class NgtPortal extends NgtSignalStore<NgtPortalInputs> implements OnInit {\n    @Input() set container(container: NgtPortalInputs['container']) {\n        this.set({ container });\n    }\n\n    @Input() set state(state: NgtPortalInputs['state']) {\n        this.set({ state });\n    }\n\n    @Input() autoRender = true;\n    @Input() autoRenderPriority = 1;\n\n    @Output() beforeRender = new EventEmitter<{ root: NgtRenderState; portal: NgtRenderState }>();\n\n    @ContentChild(NgtPortalContent, { read: TemplateRef, static: true })\n    readonly portalContentTemplate!: TemplateRef<unknown>;\n\n    @ViewChild('portalContentAnchor', { read: ViewContainerRef, static: true })\n    readonly portalContentAnchor!: ViewContainerRef;\n\n    readonly #parentStore = inject(NgtStore, { skipSelf: true });\n    readonly parentScene = this.#parentStore.get('scene');\n    readonly parentCamera = this.#parentStore.get('camera');\n\n    readonly #portalStore = inject(NgtStore, { self: true });\n    readonly #injector = inject(Injector);\n    readonly #zone = inject(NgZone);\n\n    readonly #raycaster = new THREE.Raycaster();\n    readonly #pointer = new THREE.Vector2();\n\n    portalContentRendered = false;\n    #portalContentView?: EmbeddedViewRef<unknown>;\n\n    constructor() {\n        super({ container: injectNgtRef<THREE.Scene>(prepare(new THREE.Scene())) });\n        inject(DestroyRef).onDestroy(() => {\n            if (this.#portalContentView && !this.#portalContentView.destroyed) {\n                this.#portalContentView.destroy();\n            }\n        });\n    }\n\n    ngOnInit() {\n        const previousState = this.#parentStore.get();\n        const inputsState = this.get();\n\n        if (!inputsState.state && this.autoRender) {\n            inputsState.state = { events: { priority: this.autoRenderPriority + 1 } };\n        }\n\n        const { events, size, ...restInputsState } = inputsState.state || {};\n\n        const containerState = inputsState.container;\n        const container = is.ref(containerState) ? containerState.nativeElement : containerState;\n\n        const localState = getLocalState(container);\n        if (!localState.store) {\n            localState.store = this.#portalStore;\n        }\n\n        this.#portalStore.set({\n            ...previousState,\n            scene: container as THREE.Scene,\n            raycaster: this.#raycaster,\n            pointer: this.#pointer,\n            previousStore: this.#parentStore,\n            events: { ...previousState.events, ...(events || {}) },\n            size: { ...previousState.size, ...(size || {}) },\n            ...restInputsState,\n            get: this.#portalStore.get.bind(this.#portalStore),\n            set: this.#portalStore.set.bind(this.#portalStore),\n            setEvents: (events) =>\n                this.#portalStore.set((state) => ({ ...state, events: { ...state.events, ...events } })),\n        });\n\n        const parentState = this.#parentStore.select();\n        effect(\n            () => {\n                const previous = parentState();\n                this.#zone.runOutsideAngular(() => {\n                    this.#portalStore.set((state) => this.#inject(previous, state));\n                });\n            },\n            { injector: this.#injector, allowSignalWrites: true }\n        );\n\n        requestAnimationFrame(() => {\n            this.#portalStore.set((injectState) => this.#inject(this.#parentStore.get(), injectState));\n        });\n        this.#portalContentView = this.portalContentAnchor.createEmbeddedView(this.portalContentTemplate);\n        safeDetectChanges(this.#portalContentView);\n        this.portalContentRendered = true;\n    }\n\n    onBeforeRender(portal: NgtRenderState) {\n        this.beforeRender.emit({\n            root: { ...this.#parentStore.get(), delta: portal.delta, frame: portal.frame },\n            portal,\n        });\n    }\n\n    #inject(rootState: NgtState, injectState: NgtState) {\n        const intersect: Partial<NgtState> = { ...rootState };\n\n        Object.keys(intersect).forEach((key) => {\n            if (\n                privateKeys.includes(key as PrivateKeys) ||\n                rootState[key as keyof NgtState] !== injectState[key as keyof NgtState]\n            ) {\n                delete intersect[key as keyof NgtState];\n            }\n        });\n\n        const inputs = this.get();\n        const { size, events, ...restInputsState } = inputs.state || {};\n\n        let viewport = undefined;\n        if (injectState && size) {\n            const camera = injectState.camera;\n            viewport = rootState.viewport.getCurrentViewport(camera, new THREE.Vector3(), size);\n            if (camera !== rootState.camera) updateCamera(camera, size);\n        }\n\n        return {\n            ...intersect,\n            scene: is.ref(inputs.container) ? inputs.container.nativeElement : inputs.container,\n            raycaster: this.#raycaster,\n            pointer: this.#pointer,\n            previousStore: this.#parentStore,\n            events: { ...rootState.events, ...(injectState?.events || {}), ...events },\n            size: { ...rootState.size, ...size },\n            viewport: { ...rootState.viewport, ...(viewport || {}) },\n            ...restInputsState,\n        } as NgtState;\n    }\n}\n"]}