angular-three-soba 2.1.0 → 2.2.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/abstractions/lib/catmull-rom-line.d.ts +139 -139
- package/abstractions/lib/edges.d.ts +164 -164
- package/abstractions/lib/grid.d.ts +39 -39
- package/abstractions/lib/line.d.ts +8 -7
- package/abstractions/lib/prism-geometry.d.ts +4 -4
- package/abstractions/lib/rounded-box.d.ts +3 -3
- package/abstractions/lib/text-3d.d.ts +46 -46
- package/abstractions/lib/text.d.ts +1 -2
- package/cameras/lib/cube-camera.d.ts +1 -1
- package/cameras/lib/orthographic-camera.d.ts +7 -8
- package/cameras/lib/perspective-camera.d.ts +2 -3
- package/controls/index.d.ts +9 -1
- package/controls/lib/camera-controls.d.ts +1 -1
- package/controls/lib/orbit-controls.d.ts +1 -1
- package/esm2022/abstractions/lib/billboard.mjs +3 -3
- package/esm2022/abstractions/lib/catmull-rom-line.mjs +3 -3
- package/esm2022/abstractions/lib/cubic-bezier-line.mjs +3 -3
- package/esm2022/abstractions/lib/edges.mjs +30 -34
- package/esm2022/abstractions/lib/gradient-texture.mjs +3 -3
- package/esm2022/abstractions/lib/grid.mjs +3 -3
- package/esm2022/abstractions/lib/helper.mjs +18 -20
- package/esm2022/abstractions/lib/line.mjs +25 -28
- package/esm2022/abstractions/lib/prism-geometry.mjs +3 -3
- package/esm2022/abstractions/lib/quadratic-bezier-line.mjs +3 -3
- package/esm2022/abstractions/lib/rounded-box.mjs +11 -17
- package/esm2022/abstractions/lib/text-3d.mjs +14 -17
- package/esm2022/abstractions/lib/text.mjs +11 -16
- package/esm2022/cameras/lib/camera-content.mjs +3 -3
- package/esm2022/cameras/lib/cube-camera.mjs +18 -27
- package/esm2022/cameras/lib/orthographic-camera.mjs +20 -20
- package/esm2022/cameras/lib/perspective-camera.mjs +21 -21
- package/esm2022/controls/index.mjs +6 -2
- package/esm2022/controls/lib/camera-controls.mjs +3 -3
- package/esm2022/controls/lib/orbit-controls.mjs +3 -3
- package/esm2022/controls/lib/scroll-controls.mjs +124 -130
- package/esm2022/gizmos/angular-three-soba-gizmos.mjs +5 -0
- package/esm2022/gizmos/index.mjs +6 -0
- package/esm2022/gizmos/lib/gizmo-helper/gizmo-helper.mjs +187 -0
- package/esm2022/gizmos/lib/gizmo-helper/gizmo-viewcube.mjs +304 -0
- package/esm2022/gizmos/lib/gizmo-helper/gizmo-viewport.mjs +332 -0
- package/esm2022/gizmos/lib/pivot-controls/axis-arrow.mjs +265 -0
- package/esm2022/gizmos/lib/pivot-controls/axis-rotator.mjs +264 -0
- package/esm2022/gizmos/lib/pivot-controls/pivot-controls.mjs +337 -0
- package/esm2022/gizmos/lib/pivot-controls/plane-slider.mjs +266 -0
- package/esm2022/gizmos/lib/pivot-controls/scaling-sphere.mjs +232 -0
- package/esm2022/gizmos/lib/transform-controls.mjs +177 -0
- package/esm2022/loaders/lib/loader.mjs +3 -3
- package/esm2022/materials/lib/custom-shader-material.mjs +8 -12
- package/esm2022/materials/lib/mesh-distort-material.mjs +3 -3
- package/esm2022/materials/lib/mesh-reflector-material.mjs +11 -15
- package/esm2022/materials/lib/mesh-refraction-material.mjs +26 -30
- package/esm2022/materials/lib/mesh-transmission-material.mjs +3 -3
- package/esm2022/materials/lib/mesh-wobble-material.mjs +3 -3
- package/esm2022/materials/lib/point-material.mjs +3 -3
- package/esm2022/misc/lib/bake-shadows.mjs +8 -10
- package/esm2022/misc/lib/computed-attribute.mjs +15 -19
- package/esm2022/misc/lib/decal.mjs +56 -53
- package/esm2022/misc/lib/fbo.mjs +10 -10
- package/esm2022/misc/lib/html/html-content.mjs +62 -67
- package/esm2022/misc/lib/html/html.mjs +3 -3
- package/esm2022/misc/lib/intersect.mjs +6 -9
- package/esm2022/misc/lib/sampler.mjs +48 -42
- package/esm2022/performances/lib/adaptive-dpr.mjs +29 -35
- package/esm2022/performances/lib/adaptive-events.mjs +12 -16
- package/esm2022/performances/lib/detailed.mjs +14 -18
- package/esm2022/performances/lib/instances/instances.mjs +16 -22
- package/esm2022/performances/lib/points/points.mjs +19 -25
- package/esm2022/performances/lib/segments/segments.mjs +11 -14
- package/esm2022/shaders/lib/mesh-refraction-material.mjs +1 -1
- package/esm2022/staging/index.mjs +3 -2
- package/esm2022/staging/lib/accumulative-shadows.mjs +19 -21
- package/esm2022/staging/lib/backdrop.mjs +19 -23
- package/esm2022/staging/lib/bb-anchor.mjs +3 -3
- package/esm2022/staging/lib/bounds.mjs +39 -43
- package/esm2022/staging/lib/camera-shake.mjs +14 -18
- package/esm2022/staging/lib/caustics.mjs +24 -28
- package/esm2022/staging/lib/center.mjs +28 -32
- package/esm2022/staging/lib/contact-shadows.mjs +3 -3
- package/esm2022/staging/lib/environment/environment.mjs +338 -0
- package/esm2022/staging/lib/environment/inject-environment.mjs +184 -0
- package/esm2022/staging/lib/float.mjs +3 -3
- package/esm2022/staging/lib/lightformer.mjs +18 -22
- package/esm2022/staging/lib/mask.mjs +8 -12
- package/esm2022/staging/lib/matcap-texture.mjs +9 -11
- package/esm2022/staging/lib/normal-texture.mjs +9 -11
- package/esm2022/staging/lib/randomized-lights.mjs +9 -13
- package/esm2022/staging/lib/render-texture.mjs +19 -20
- package/esm2022/staging/lib/sky.mjs +3 -3
- package/esm2022/staging/lib/spot-light.mjs +32 -40
- package/esm2022/staging/lib/stage.mjs +13 -17
- package/esm2022/stats/lib/stats.mjs +21 -27
- package/fesm2022/angular-three-soba-abstractions.mjs +119 -138
- package/fesm2022/angular-three-soba-abstractions.mjs.map +1 -1
- package/fesm2022/angular-three-soba-cameras.mjs +57 -64
- package/fesm2022/angular-three-soba-cameras.mjs.map +1 -1
- package/fesm2022/angular-three-soba-controls.mjs +141 -1470
- package/fesm2022/angular-three-soba-controls.mjs.map +1 -1
- package/fesm2022/angular-three-soba-gizmos.mjs +2318 -0
- package/fesm2022/angular-three-soba-gizmos.mjs.map +1 -0
- package/fesm2022/angular-three-soba-loaders.mjs +3 -3
- package/fesm2022/angular-three-soba-materials.mjs +52 -62
- package/fesm2022/angular-three-soba-materials.mjs.map +1 -1
- package/fesm2022/angular-three-soba-misc.mjs +194 -195
- package/fesm2022/angular-three-soba-misc.mjs.map +1 -1
- package/fesm2022/angular-three-soba-performances.mjs +90 -114
- package/fesm2022/angular-three-soba-performances.mjs.map +1 -1
- package/fesm2022/angular-three-soba-shaders.mjs.map +1 -1
- package/fesm2022/angular-three-soba-staging.mjs +427 -430
- package/fesm2022/angular-three-soba-staging.mjs.map +1 -1
- package/fesm2022/angular-three-soba-stats.mjs +20 -26
- package/fesm2022/angular-three-soba-stats.mjs.map +1 -1
- package/gizmos/README.md +3 -0
- package/gizmos/index.d.ts +5 -0
- package/gizmos/lib/gizmo-helper/gizmo-helper.d.ts +45 -0
- package/gizmos/lib/gizmo-helper/gizmo-viewcube.d.ts +74 -0
- package/gizmos/lib/gizmo-helper/gizmo-viewport.d.ts +64 -0
- package/{controls → gizmos}/lib/pivot-controls/pivot-controls.d.ts +1 -1
- package/gizmos/lib/transform-controls.d.ts +50 -0
- package/materials/lib/mesh-transmission-material.d.ts +1 -1
- package/misc/lib/computed-attribute.d.ts +2 -2
- package/misc/lib/decal.d.ts +4 -4
- package/misc/lib/html/html-content.d.ts +6 -6
- package/misc/lib/html/html.d.ts +1 -1
- package/misc/lib/sampler.d.ts +1 -3
- package/package.json +14 -8
- package/performances/lib/instances/instances.d.ts +3 -3
- package/shaders/lib/mesh-refraction-material.d.ts +2 -5
- package/staging/index.d.ts +2 -1
- package/staging/lib/camera-shake.d.ts +0 -1
- package/staging/lib/caustics.d.ts +1 -1
- package/staging/lib/center.d.ts +2 -2
- package/staging/lib/contact-shadows.d.ts +1 -1
- package/staging/lib/{environment.d.ts → environment/environment.d.ts} +20 -49
- package/staging/lib/environment/inject-environment.d.ts +33 -0
- package/staging/lib/lightformer.d.ts +1 -1
- package/staging/lib/mask.d.ts +1 -1
- package/staging/lib/matcap-texture.d.ts +0 -1
- package/staging/lib/normal-texture.d.ts +0 -1
- package/staging/lib/render-texture.d.ts +1 -1
- package/staging/lib/sky.d.ts +1 -1
- package/staging/lib/stage.d.ts +3 -2
- package/esm2022/controls/lib/pivot-controls/axis-arrow.mjs +0 -263
- package/esm2022/controls/lib/pivot-controls/axis-rotator.mjs +0 -264
- package/esm2022/controls/lib/pivot-controls/pivot-controls.mjs +0 -340
- package/esm2022/controls/lib/pivot-controls/plane-slider.mjs +0 -266
- package/esm2022/controls/lib/pivot-controls/scaling-sphere.mjs +0 -232
- package/esm2022/staging/lib/environment.mjs +0 -481
- /package/{controls → gizmos}/lib/pivot-controls/axis-arrow.d.ts +0 -0
- /package/{controls → gizmos}/lib/pivot-controls/axis-rotator.d.ts +0 -0
- /package/{controls → gizmos}/lib/pivot-controls/plane-slider.d.ts +0 -0
- /package/{controls → gizmos}/lib/pivot-controls/scaling-sphere.d.ts +0 -0
|
@@ -0,0 +1,338 @@
|
|
|
1
|
+
import { NgTemplateOutlet } from '@angular/common';
|
|
2
|
+
import { ChangeDetectionStrategy, Component, computed, contentChild, CUSTOM_ELEMENTS_SCHEMA, DestroyRef, Directive, effect, inject, Injector, input, output, TemplateRef, viewChild, } from '@angular/core';
|
|
3
|
+
import { applyProps, extend, injectBeforeRender, injectStore, is, NgtArgs, NgtPortal, NgtPortalContent, pick, prepare, } from 'angular-three';
|
|
4
|
+
import { mergeInputs } from 'ngxtension/inject-inputs';
|
|
5
|
+
import { CubeCamera, HalfFloatType, Scene, WebGLCubeRenderTarget } from 'three';
|
|
6
|
+
import { GroundProjectedEnv } from 'three-stdlib';
|
|
7
|
+
import { injectEnvironment } from './inject-environment';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
function resolveScene(scene) {
|
|
10
|
+
return is.ref(scene) ? scene.nativeElement : scene;
|
|
11
|
+
}
|
|
12
|
+
function setEnvProps(background, scene, defaultScene, texture, sceneProps = {}) {
|
|
13
|
+
sceneProps.backgroundBlurriness ??= sceneProps.blur ?? 0;
|
|
14
|
+
sceneProps.backgroundIntensity ??= 1;
|
|
15
|
+
// @ts-expect-error - it's ok, we're sending through applyProps
|
|
16
|
+
sceneProps.backgroundRotation ??= [0, 0, 0];
|
|
17
|
+
sceneProps.environmentIntensity ??= 1;
|
|
18
|
+
// @ts-expect-error - it's ok, we're sending through applyProps
|
|
19
|
+
sceneProps.environmentRotation ??= [0, 0, 0];
|
|
20
|
+
const target = resolveScene(scene || defaultScene);
|
|
21
|
+
const oldbg = target.background;
|
|
22
|
+
const oldenv = target.environment;
|
|
23
|
+
const oldSceneProps = {
|
|
24
|
+
backgroundBlurriness: target.backgroundBlurriness,
|
|
25
|
+
backgroundIntensity: target.backgroundIntensity,
|
|
26
|
+
backgroundRotation: target.backgroundRotation?.clone?.() ?? [0, 0, 0],
|
|
27
|
+
environmentIntensity: target.environmentIntensity,
|
|
28
|
+
environmentRotation: target.environmentRotation?.clone?.() ?? [0, 0, 0],
|
|
29
|
+
};
|
|
30
|
+
if (background !== 'only')
|
|
31
|
+
target.environment = texture;
|
|
32
|
+
if (background)
|
|
33
|
+
target.background = texture;
|
|
34
|
+
applyProps(target, sceneProps);
|
|
35
|
+
return () => {
|
|
36
|
+
if (background !== 'only')
|
|
37
|
+
target.environment = oldenv;
|
|
38
|
+
if (background)
|
|
39
|
+
target.background = oldbg;
|
|
40
|
+
applyProps(target, oldSceneProps);
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
const defaultBackground = {
|
|
44
|
+
background: false,
|
|
45
|
+
};
|
|
46
|
+
export class NgtsEnvironmentMap {
|
|
47
|
+
constructor() {
|
|
48
|
+
this.options = input(defaultBackground, { transform: mergeInputs(defaultBackground) });
|
|
49
|
+
this.envSet = output();
|
|
50
|
+
const store = injectStore();
|
|
51
|
+
const defaultScene = store.select('scene');
|
|
52
|
+
const _map = pick(this.options, 'map');
|
|
53
|
+
const _envConfig = computed(() => {
|
|
54
|
+
const { background = false, scene, blur, backgroundBlurriness, backgroundIntensity, backgroundRotation, environmentIntensity, environmentRotation, } = this.options();
|
|
55
|
+
return {
|
|
56
|
+
background,
|
|
57
|
+
scene,
|
|
58
|
+
blur,
|
|
59
|
+
backgroundBlurriness,
|
|
60
|
+
backgroundIntensity,
|
|
61
|
+
backgroundRotation,
|
|
62
|
+
environmentIntensity,
|
|
63
|
+
environmentRotation,
|
|
64
|
+
};
|
|
65
|
+
});
|
|
66
|
+
effect((onCleanup) => {
|
|
67
|
+
const map = _map();
|
|
68
|
+
if (!map)
|
|
69
|
+
return;
|
|
70
|
+
const { background = false, scene, ...config } = _envConfig();
|
|
71
|
+
const cleanup = setEnvProps(background, scene, defaultScene(), map, config);
|
|
72
|
+
this.envSet.emit();
|
|
73
|
+
onCleanup(() => cleanup());
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironmentMap, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
77
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.4", type: NgtsEnvironmentMap, isStandalone: true, selector: "ngts-environment-map", inputs: { options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { envSet: "envSet" }, ngImport: i0 }); }
|
|
78
|
+
}
|
|
79
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironmentMap, decorators: [{
|
|
80
|
+
type: Directive,
|
|
81
|
+
args: [{ standalone: true, selector: 'ngts-environment-map' }]
|
|
82
|
+
}], ctorParameters: () => [] });
|
|
83
|
+
export class NgtsEnvironmentCube {
|
|
84
|
+
constructor() {
|
|
85
|
+
this.options = input(defaultBackground, { transform: mergeInputs(defaultBackground) });
|
|
86
|
+
this.envSet = output();
|
|
87
|
+
this.store = injectStore();
|
|
88
|
+
this.defaultScene = this.store.select('scene');
|
|
89
|
+
this.envConfig = computed(() => {
|
|
90
|
+
const { background = false, scene, blur, backgroundBlurriness, backgroundIntensity, backgroundRotation, environmentIntensity, environmentRotation, } = this.options();
|
|
91
|
+
return {
|
|
92
|
+
background,
|
|
93
|
+
scene,
|
|
94
|
+
blur,
|
|
95
|
+
backgroundBlurriness,
|
|
96
|
+
backgroundIntensity,
|
|
97
|
+
backgroundRotation,
|
|
98
|
+
environmentIntensity,
|
|
99
|
+
environmentRotation,
|
|
100
|
+
};
|
|
101
|
+
});
|
|
102
|
+
const _texture = injectEnvironment(this.options);
|
|
103
|
+
effect((onCleanup) => {
|
|
104
|
+
const texture = _texture();
|
|
105
|
+
if (!texture)
|
|
106
|
+
return;
|
|
107
|
+
const { background = false, scene, ...config } = this.envConfig();
|
|
108
|
+
const cleanup = setEnvProps(background, scene, this.defaultScene(), texture, config);
|
|
109
|
+
this.envSet.emit();
|
|
110
|
+
onCleanup(() => cleanup());
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironmentCube, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
114
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.4", type: NgtsEnvironmentCube, isStandalone: true, selector: "ngts-environment-cube", inputs: { options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { envSet: "envSet" }, ngImport: i0 }); }
|
|
115
|
+
}
|
|
116
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironmentCube, decorators: [{
|
|
117
|
+
type: Directive,
|
|
118
|
+
args: [{ standalone: true, selector: 'ngts-environment-cube' }]
|
|
119
|
+
}], ctorParameters: () => [] });
|
|
120
|
+
export class NgtsEnvironmentPortal {
|
|
121
|
+
constructor() {
|
|
122
|
+
this.defaultOptions = {
|
|
123
|
+
near: 1,
|
|
124
|
+
far: 1000,
|
|
125
|
+
resolution: 256,
|
|
126
|
+
frames: 1,
|
|
127
|
+
background: false,
|
|
128
|
+
};
|
|
129
|
+
this.options = input(this.defaultOptions, { transform: mergeInputs(this.defaultOptions) });
|
|
130
|
+
this.content = input.required();
|
|
131
|
+
this.envSet = output();
|
|
132
|
+
this.injector = inject(Injector);
|
|
133
|
+
this.store = injectStore();
|
|
134
|
+
this.defaultScene = this.store.select('scene');
|
|
135
|
+
this.gl = this.store.select('gl');
|
|
136
|
+
this.cameraRef = viewChild('cubeCamera');
|
|
137
|
+
this.map = pick(this.options, 'map');
|
|
138
|
+
this.files = pick(this.options, 'files');
|
|
139
|
+
this.preset = pick(this.options, 'preset');
|
|
140
|
+
this.extensions = pick(this.options, 'extensions');
|
|
141
|
+
this.path = pick(this.options, 'path');
|
|
142
|
+
this.envMapOptions = computed(() => ({ background: true, map: this.map(), extensions: this.extensions() }));
|
|
143
|
+
this.envCubeOptions = computed(() => ({
|
|
144
|
+
background: true,
|
|
145
|
+
files: this.files(),
|
|
146
|
+
preset: this.preset(),
|
|
147
|
+
extensions: this.extensions(),
|
|
148
|
+
path: this.path(),
|
|
149
|
+
}));
|
|
150
|
+
this.near = pick(this.options, 'near');
|
|
151
|
+
this.far = pick(this.options, 'far');
|
|
152
|
+
this.resolution = pick(this.options, 'resolution');
|
|
153
|
+
this.fbo = computed(() => {
|
|
154
|
+
const fbo = new WebGLCubeRenderTarget(this.resolution());
|
|
155
|
+
fbo.texture.type = HalfFloatType;
|
|
156
|
+
return fbo;
|
|
157
|
+
});
|
|
158
|
+
this.cameraArgs = computed(() => [this.near(), this.far(), this.fbo()]);
|
|
159
|
+
this.virtualScene = prepare(new Scene());
|
|
160
|
+
extend({ CubeCamera });
|
|
161
|
+
effect((onCleanup) => {
|
|
162
|
+
const [files, preset, map] = [this.files(), this.preset(), this.map()];
|
|
163
|
+
// NOTE: when there's none of this, we don't render cube or map so we need to setEnv here
|
|
164
|
+
if (!!files || !!preset || !!map)
|
|
165
|
+
return;
|
|
166
|
+
const cleanup = this.setPortalEnv();
|
|
167
|
+
onCleanup(() => cleanup?.());
|
|
168
|
+
});
|
|
169
|
+
let count = 1;
|
|
170
|
+
injectBeforeRender(() => {
|
|
171
|
+
const frames = this.options().frames;
|
|
172
|
+
if (frames === Infinity || (frames != null && count < frames)) {
|
|
173
|
+
const camera = this.cameraRef()?.nativeElement;
|
|
174
|
+
if (camera) {
|
|
175
|
+
camera.update(this.gl(), this.virtualScene);
|
|
176
|
+
count++;
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
});
|
|
180
|
+
inject(DestroyRef).onDestroy(() => {
|
|
181
|
+
if (this.setEnvEffectRef)
|
|
182
|
+
this.setEnvEffectRef.destroy();
|
|
183
|
+
});
|
|
184
|
+
}
|
|
185
|
+
// NOTE: we use onEnvSet here to ensure that EnvironmentCube or EnvironmentMap sets the env before the portal
|
|
186
|
+
onEnvSet() {
|
|
187
|
+
if (this.setEnvEffectRef)
|
|
188
|
+
this.setEnvEffectRef.destroy();
|
|
189
|
+
this.setEnvEffectRef = effect((onCleanup) => {
|
|
190
|
+
const cleanup = this.setPortalEnv();
|
|
191
|
+
onCleanup(() => cleanup?.());
|
|
192
|
+
}, { manualCleanup: true, injector: this.injector });
|
|
193
|
+
}
|
|
194
|
+
setPortalEnv() {
|
|
195
|
+
const camera = this.cameraRef();
|
|
196
|
+
if (!camera?.nativeElement)
|
|
197
|
+
return;
|
|
198
|
+
const [{ frames, background = false, scene, blur, backgroundBlurriness, backgroundIntensity, backgroundRotation, environmentIntensity, environmentRotation, }, gl, fbo, defaultScene,] = [this.options(), this.gl(), this.fbo(), this.defaultScene()];
|
|
199
|
+
if (frames === 1)
|
|
200
|
+
camera.nativeElement.update(gl, this.virtualScene);
|
|
201
|
+
const cleanup = setEnvProps(background, scene, defaultScene, fbo.texture, {
|
|
202
|
+
blur,
|
|
203
|
+
backgroundBlurriness,
|
|
204
|
+
backgroundIntensity,
|
|
205
|
+
backgroundRotation,
|
|
206
|
+
environmentIntensity,
|
|
207
|
+
environmentRotation,
|
|
208
|
+
});
|
|
209
|
+
this.envSet.emit();
|
|
210
|
+
return cleanup;
|
|
211
|
+
}
|
|
212
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironmentPortal, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
213
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.4", type: NgtsEnvironmentPortal, isStandalone: true, selector: "ngts-environment-portal", inputs: { options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: false, transformFunction: null }, content: { classPropertyName: "content", publicName: "content", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { envSet: "envSet" }, viewQueries: [{ propertyName: "cameraRef", first: true, predicate: ["cubeCamera"], descendants: true, isSignal: true }], ngImport: i0, template: `
|
|
214
|
+
<ngt-portal [container]="virtualScene">
|
|
215
|
+
<ng-template portalContent let-injector="injector" let-container="container">
|
|
216
|
+
<ng-container
|
|
217
|
+
[ngTemplateOutlet]="content()"
|
|
218
|
+
[ngTemplateOutletInjector]="injector"
|
|
219
|
+
[ngTemplateOutletContext]="{ injector, container }"
|
|
220
|
+
/>
|
|
221
|
+
|
|
222
|
+
<ngt-cube-camera #cubeCamera *args="cameraArgs()" />
|
|
223
|
+
|
|
224
|
+
@if (files() || preset()) {
|
|
225
|
+
<ngts-environment-cube [options]="envCubeOptions()" (envSet)="onEnvSet()" />
|
|
226
|
+
} @else if (map()) {
|
|
227
|
+
<ngts-environment-map [options]="envMapOptions()" (envSet)="onEnvSet()" />
|
|
228
|
+
}
|
|
229
|
+
</ng-template>
|
|
230
|
+
</ngt-portal>
|
|
231
|
+
`, isInline: true, dependencies: [{ kind: "directive", type: NgtsEnvironmentCube, selector: "ngts-environment-cube", inputs: ["options"], outputs: ["envSet"] }, { kind: "directive", type: NgtsEnvironmentMap, selector: "ngts-environment-map", inputs: ["options"], outputs: ["envSet"] }, { kind: "directive", type: NgtArgs, selector: "ng-template[args]", inputs: ["args"] }, { kind: "component", type: NgtPortal, selector: "ngt-portal", inputs: ["container", "state", "autoRender", "autoRenderPriority"] }, { kind: "directive", type: NgtPortalContent, selector: "ng-template[portalContent]" }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
232
|
+
}
|
|
233
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironmentPortal, decorators: [{
|
|
234
|
+
type: Component,
|
|
235
|
+
args: [{
|
|
236
|
+
selector: 'ngts-environment-portal',
|
|
237
|
+
standalone: true,
|
|
238
|
+
template: `
|
|
239
|
+
<ngt-portal [container]="virtualScene">
|
|
240
|
+
<ng-template portalContent let-injector="injector" let-container="container">
|
|
241
|
+
<ng-container
|
|
242
|
+
[ngTemplateOutlet]="content()"
|
|
243
|
+
[ngTemplateOutletInjector]="injector"
|
|
244
|
+
[ngTemplateOutletContext]="{ injector, container }"
|
|
245
|
+
/>
|
|
246
|
+
|
|
247
|
+
<ngt-cube-camera #cubeCamera *args="cameraArgs()" />
|
|
248
|
+
|
|
249
|
+
@if (files() || preset()) {
|
|
250
|
+
<ngts-environment-cube [options]="envCubeOptions()" (envSet)="onEnvSet()" />
|
|
251
|
+
} @else if (map()) {
|
|
252
|
+
<ngts-environment-map [options]="envMapOptions()" (envSet)="onEnvSet()" />
|
|
253
|
+
}
|
|
254
|
+
</ng-template>
|
|
255
|
+
</ngt-portal>
|
|
256
|
+
`,
|
|
257
|
+
imports: [NgtsEnvironmentCube, NgtsEnvironmentMap, NgtArgs, NgtPortal, NgtPortalContent, NgTemplateOutlet],
|
|
258
|
+
schemas: [CUSTOM_ELEMENTS_SCHEMA],
|
|
259
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
260
|
+
}]
|
|
261
|
+
}], ctorParameters: () => [] });
|
|
262
|
+
export class NgtsEnvironmentGround {
|
|
263
|
+
constructor() {
|
|
264
|
+
this.options = input({});
|
|
265
|
+
this.envSet = output();
|
|
266
|
+
this.defaultTexture = injectEnvironment(this.options);
|
|
267
|
+
this.height = computed(() => this.options().ground?.height);
|
|
268
|
+
this.radius = computed(() => this.options().ground?.radius);
|
|
269
|
+
this.scale = computed(() => this.options().ground?.scale ?? 1000);
|
|
270
|
+
this.args = computed(() => [this.options().map || this.defaultTexture()]);
|
|
271
|
+
this.envMapOptions = computed(() => {
|
|
272
|
+
const { map: _, ...options } = this.options();
|
|
273
|
+
const [map] = this.args();
|
|
274
|
+
return Object.assign(options, { map });
|
|
275
|
+
});
|
|
276
|
+
extend({ GroundProjectedEnv });
|
|
277
|
+
}
|
|
278
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironmentGround, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
279
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.4", type: NgtsEnvironmentGround, isStandalone: true, selector: "ngts-environment-ground", inputs: { options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { envSet: "envSet" }, ngImport: i0, template: `
|
|
280
|
+
<ngts-environment-map [options]="envMapOptions()" (envSet)="envSet.emit()" />
|
|
281
|
+
<ngt-ground-projected-env *args="args()" [scale]="scale()" [height]="height()" [radius]="radius()" />
|
|
282
|
+
`, isInline: true, dependencies: [{ kind: "directive", type: NgtsEnvironmentMap, selector: "ngts-environment-map", inputs: ["options"], outputs: ["envSet"] }, { kind: "directive", type: NgtArgs, selector: "ng-template[args]", inputs: ["args"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
283
|
+
}
|
|
284
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironmentGround, decorators: [{
|
|
285
|
+
type: Component,
|
|
286
|
+
args: [{
|
|
287
|
+
selector: 'ngts-environment-ground',
|
|
288
|
+
standalone: true,
|
|
289
|
+
template: `
|
|
290
|
+
<ngts-environment-map [options]="envMapOptions()" (envSet)="envSet.emit()" />
|
|
291
|
+
<ngt-ground-projected-env *args="args()" [scale]="scale()" [height]="height()" [radius]="radius()" />
|
|
292
|
+
`,
|
|
293
|
+
imports: [NgtsEnvironmentMap, NgtArgs],
|
|
294
|
+
schemas: [CUSTOM_ELEMENTS_SCHEMA],
|
|
295
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
296
|
+
}]
|
|
297
|
+
}], ctorParameters: () => [] });
|
|
298
|
+
export class NgtsEnvironment {
|
|
299
|
+
constructor() {
|
|
300
|
+
this.options = input({});
|
|
301
|
+
this.content = contentChild(TemplateRef);
|
|
302
|
+
this.envSet = output();
|
|
303
|
+
}
|
|
304
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironment, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
305
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.4", type: NgtsEnvironment, isStandalone: true, selector: "ngts-environment", inputs: { options: { classPropertyName: "options", publicName: "options", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { envSet: "envSet" }, queries: [{ propertyName: "content", first: true, predicate: TemplateRef, descendants: true, isSignal: true }], ngImport: i0, template: `
|
|
306
|
+
@if (options().ground) {
|
|
307
|
+
<ngts-environment-ground [options]="options()" (envSet)="envSet.emit()" />
|
|
308
|
+
} @else if (options().map) {
|
|
309
|
+
<ngts-environment-map [options]="options()" (envSet)="envSet.emit()" />
|
|
310
|
+
} @else if (content()) {
|
|
311
|
+
<ngts-environment-portal [options]="options()" [content]="$any(content())" (envSet)="envSet.emit()" />
|
|
312
|
+
} @else {
|
|
313
|
+
<ngts-environment-cube [options]="options()" (envSet)="envSet.emit()" />
|
|
314
|
+
}
|
|
315
|
+
`, isInline: true, dependencies: [{ kind: "directive", type: NgtsEnvironmentCube, selector: "ngts-environment-cube", inputs: ["options"], outputs: ["envSet"] }, { kind: "directive", type: NgtsEnvironmentMap, selector: "ngts-environment-map", inputs: ["options"], outputs: ["envSet"] }, { kind: "component", type: NgtsEnvironmentPortal, selector: "ngts-environment-portal", inputs: ["options", "content"], outputs: ["envSet"] }, { kind: "component", type: NgtsEnvironmentGround, selector: "ngts-environment-ground", inputs: ["options"], outputs: ["envSet"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
316
|
+
}
|
|
317
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.4", ngImport: i0, type: NgtsEnvironment, decorators: [{
|
|
318
|
+
type: Component,
|
|
319
|
+
args: [{
|
|
320
|
+
selector: 'ngts-environment',
|
|
321
|
+
standalone: true,
|
|
322
|
+
template: `
|
|
323
|
+
@if (options().ground) {
|
|
324
|
+
<ngts-environment-ground [options]="options()" (envSet)="envSet.emit()" />
|
|
325
|
+
} @else if (options().map) {
|
|
326
|
+
<ngts-environment-map [options]="options()" (envSet)="envSet.emit()" />
|
|
327
|
+
} @else if (content()) {
|
|
328
|
+
<ngts-environment-portal [options]="options()" [content]="$any(content())" (envSet)="envSet.emit()" />
|
|
329
|
+
} @else {
|
|
330
|
+
<ngts-environment-cube [options]="options()" (envSet)="envSet.emit()" />
|
|
331
|
+
}
|
|
332
|
+
`,
|
|
333
|
+
imports: [NgtsEnvironmentCube, NgtsEnvironmentMap, NgtsEnvironmentPortal, NgtsEnvironmentGround],
|
|
334
|
+
schemas: [CUSTOM_ELEMENTS_SCHEMA],
|
|
335
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
336
|
+
}]
|
|
337
|
+
}] });
|
|
338
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW52aXJvbm1lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL3NvYmEvc3RhZ2luZy9zcmMvbGliL2Vudmlyb25tZW50L2Vudmlyb25tZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ25ELE9BQU8sRUFDTix1QkFBdUIsRUFDdkIsU0FBUyxFQUNULFFBQVEsRUFDUixZQUFZLEVBQ1osc0JBQXNCLEVBQ3RCLFVBQVUsRUFDVixTQUFTLEVBQ1QsTUFBTSxFQUdOLE1BQU0sRUFDTixRQUFRLEVBQ1IsS0FBSyxFQUNMLE1BQU0sRUFDTixXQUFXLEVBQ1gsU0FBUyxHQUNULE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFDTixVQUFVLEVBQ1YsTUFBTSxFQUNOLGtCQUFrQixFQUNsQixXQUFXLEVBQ1gsRUFBRSxFQUNGLE9BQU8sRUFDUCxTQUFTLEVBQ1QsZ0JBQWdCLEVBQ2hCLElBQUksRUFDSixPQUFPLEdBQ1AsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ3ZELE9BQU8sRUFBRSxVQUFVLEVBQVMsYUFBYSxFQUFFLEtBQUssRUFBVyxxQkFBcUIsRUFBRSxNQUFNLE9BQU8sQ0FBQztBQUNoRyxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDbEQsT0FBTyxFQUFFLGlCQUFpQixFQUF3RCxNQUFNLHNCQUFzQixDQUFDOztBQUUvRyxTQUFTLFlBQVksQ0FBQyxLQUFnQztJQUNyRCxPQUFPLEVBQUUsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxhQUFhLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQztBQUNwRCxDQUFDO0FBRUQsU0FBUyxXQUFXLENBQ25CLFVBQTRCLEVBQzVCLEtBQTRD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QU8sZUFBZTtJQWxCNUI7UUFtQkMsWUFBTyxHQUFHLEtBQUssQ0FBQyxFQUE0QixDQUFDLENBQUM7UUFDOUMsWUFBTyxHQUFHLFlBQVksQ0FBQyxXQUFXLENBQUMsQ0FBQztRQUNwQyxXQUFNLEdBQUcsTUFBTSxFQUFRLENBQUM7S0FDeEI7OEdBSlksZUFBZTtrR0FBZixlQUFlLDBSQUVKLFdBQVcsZ0VBakJ4Qjs7Ozs7Ozs7OztFQVVULDREQS9PVyxtQkFBbUIsNEdBN0NuQixrQkFBa0IsMkdBb0hsQixxQkFBcUIseUhBc0lyQixxQkFBcUI7OzJGQXVDckIsZUFBZTtrQkFsQjNCLFNBQVM7bUJBQUM7b0JBQ1YsUUFBUSxFQUFFLGtCQUFrQjtvQkFDNUIsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLFFBQVEsRUFBRTs7Ozs7Ozs7OztFQVVUO29CQUNELE9BQU8sRUFBRSxDQUFDLG1CQUFtQixFQUFFLGtCQUFrQixFQUFFLHFCQUFxQixFQUFFLHFCQUFxQixDQUFDO29CQUNoRyxPQUFPLEVBQUUsQ0FBQyxzQkFBc0IsQ0FBQztvQkFDakMsZUFBZSxFQUFFLHVCQUF1QixDQUFDLE1BQU07aUJBQy9DIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdUZW1wbGF0ZU91dGxldCB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQge1xuXHRDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSxcblx0Q29tcG9uZW50LFxuXHRjb21wdXRlZCxcblx0Y29udGVudENoaWxkLFxuXHRDVVNUT01fRUxFTUVOVFNfU0NIRU1BLFxuXHREZXN0cm95UmVmLFxuXHREaXJlY3RpdmUsXG5cdGVmZmVjdCxcblx0RWZmZWN0UmVmLFxuXHRFbGVtZW50UmVmLFxuXHRpbmplY3QsXG5cdEluamVjdG9yLFxuXHRpbnB1dCxcblx0b3V0cHV0LFxuXHRUZW1wbGF0ZVJlZixcblx0dmlld0NoaWxkLFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7XG5cdGFwcGx5UHJvcHMsXG5cdGV4dGVuZCxcblx0aW5qZWN0QmVmb3JlUmVuZGVyLFxuXHRpbmplY3RTdG9yZSxcblx0aXMsXG5cdE5ndEFyZ3MsXG5cdE5ndFBvcnRhbCxcblx0Tmd0UG9ydGFsQ29udGVudCxcblx0cGljayxcblx0cHJlcGFyZSxcbn0gZnJvbSAnYW5ndWxhci10aHJlZSc7XG5pbXBvcnQgeyBtZXJnZUlucHV0cyB9IGZyb20gJ25neHRlbnNpb24vaW5qZWN0LWlucHV0cyc7XG5pbXBvcnQgeyBDdWJlQ2FtZXJhLCBFdWxlciwgSGFsZkZsb2F0VHlwZSwgU2NlbmUsIFRleHR1cmUsIFdlYkdMQ3ViZVJlbmRlclRhcmdldCB9IGZyb20gJ3RocmVlJztcbmltcG9ydCB7IEdyb3VuZFByb2plY3RlZEVudiB9IGZyb20gJ3RocmVlLXN0ZGxpYic7XG5pbXBvcnQgeyBpbmplY3RFbnZpcm9ubWVudCwgTmd0c0Vudmlyb25tZW50UHJlc2V0cywgTmd0c0luamVjdEVudmlyb25tZW50T3B0aW9ucyB9IGZyb20gJy4vaW5qZWN0LWVudmlyb25tZW50JztcblxuZnVuY3Rpb24gcmVzb2x2ZVNjZW5lKHNjZW5lOiBTY2VuZSB8IEVsZW1lbnRSZWY8U2NlbmU+KSB7XG5cdHJldHVybiBpcy5yZWYoc2NlbmUpID8gc2NlbmUubmF0aXZlRWxlbWVudCA6IHNjZW5lO1xufVxuXG5mdW5jdGlvbiBzZXRFbnZQcm9wcyhcblx0YmFja2dyb3VuZDogYm9vbGVhbiB8ICdvbmx5Jyxcblx0c2NlbmU6IFNjZW5lIHwgRWxlbWVudFJlZjxTY2VuZT4gfCB1bmRlZmluZWQsXG5cdGRlZmF1bHRTY2VuZTogU2NlbmUsXG5cdHRleHR1cmU6IFRleHR1cmUsXG5cdHNjZW5lUHJvcHM6IFBhcnRpYWw8Tmd0c0Vudmlyb25tZW50T3B0aW9ucz4gPSB7fSxcbikge1xuXHRzY2VuZVByb3BzLmJhY2tncm91bmRCbHVycmluZXNzID8/PSBzY2VuZVByb3BzLmJsdXIgPz8gMDtcblx0c2NlbmVQcm9wcy5iYWNrZ3JvdW5kSW50ZW5zaXR5ID8/PSAxO1xuXHQvLyBAdHMtZXhwZWN0LWVycm9yIC0gaXQncyBvaywgd2UncmUgc2VuZGluZyB0aHJvdWdoIGFwcGx5UHJvcHNcblx0c2NlbmVQcm9wcy5iYWNrZ3JvdW5kUm90YXRpb24gPz89IFswLCAwLCAwXTtcblx0c2NlbmVQcm9wcy5lbnZpcm9ubWVudEludGVuc2l0eSA/Pz0gMTtcblx0Ly8gQHRzLWV4cGVjdC1lcnJvciAtIGl0J3Mgb2ssIHdlJ3JlIHNlbmRpbmcgdGhyb3VnaCBhcHBseVByb3BzXG5cdHNjZW5lUHJvcHMuZW52aXJvbm1lbnRSb3RhdGlvbiA/Pz0gWzAsIDAsIDBdO1xuXG5cdGNvbnN0IHRhcmdldCA9IHJlc29sdmVTY2VuZShzY2VuZSB8fCBkZWZhdWx0U2NlbmUpO1xuXHRjb25zdCBvbGRiZyA9IHRhcmdldC5iYWNrZ3JvdW5kO1xuXHRjb25zdCBvbGRlbnYgPSB0YXJnZXQuZW52aXJvbm1lbnQ7XG5cdGNvbnN0IG9sZFNjZW5lUHJvcHMgPSB7XG5cdFx0YmFja2dyb3VuZEJsdXJyaW5lc3M6IHRhcmdldC5iYWNrZ3JvdW5kQmx1cnJpbmVzcyxcblx0XHRiYWNrZ3JvdW5kSW50ZW5zaXR5OiB0YXJnZXQuYmFja2dyb3VuZEludGVuc2l0eSxcblx0XHRiYWNrZ3JvdW5kUm90YXRpb246IHRhcmdldC5iYWNrZ3JvdW5kUm90YXRpb24/LmNsb25lPy4oKSA/PyBbMCwgMCwgMF0sXG5cdFx0ZW52aXJvbm1lbnRJbnRlbnNpdHk6IHRhcmdldC5lbnZpcm9ubWVudEludGVuc2l0eSxcblx0XHRlbnZpcm9ubWVudFJvdGF0aW9uOiB0YXJnZXQuZW52aXJvbm1lbnRSb3RhdGlvbj8uY2xvbmU/LigpID8/IFswLCAwLCAwXSxcblx0fTtcblxuXHRpZiAoYmFja2dyb3VuZCAhPT0gJ29ubHknKSB0YXJnZXQuZW52aXJvbm1lbnQgPSB0ZXh0dXJlO1xuXHRpZiAoYmFja2dyb3VuZCkgdGFyZ2V0LmJhY2tncm91bmQgPSB0ZXh0dXJlO1xuXHRhcHBseVByb3BzKHRhcmdldCwgc2NlbmVQcm9wcyk7XG5cblx0cmV0dXJuICgpID0+IHtcblx0XHRpZiAoYmFja2dyb3VuZCAhPT0gJ29ubHknKSB0YXJnZXQuZW52aXJvbm1lbnQgPSBvbGRlbnY7XG5cdFx0aWYgKGJhY2tncm91bmQpIHRhcmdldC5iYWNrZ3JvdW5kID0gb2xkYmc7XG5cdFx0YXBwbHlQcm9wcyh0YXJnZXQsIG9sZFNjZW5lUHJvcHMpO1xuXHR9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIE5ndHNFbnZpcm9ubWVudE9wdGlvbnMgZXh0ZW5kcyBQYXJ0aWFsPE5ndHNJbmplY3RFbnZpcm9ubWVudE9wdGlvbnM+IHtcblx0ZnJhbWVzPzogbnVtYmVyO1xuXHRuZWFyPzogbnVtYmVyO1xuXHRmYXI/OiBudW1iZXI7XG5cdHJlc29sdXRpb24/OiBudW1iZXI7XG5cdGJhY2tncm91bmQ/OiBib29sZWFuIHwgJ29ubHknO1xuXG5cdC8qKiBkZXByZWNhdGVkLCB1c2UgYmFja2dyb3VuZEJsdXJyaW5lc3MgKi9cblx0Ymx1cj86IG51bWJlcjtcblx0YmFja2dyb3VuZEJsdXJyaW5lc3M/OiBudW1iZXI7XG5cdGJhY2tncm91bmRJbnRlbnNpdHk/OiBudW1iZXI7XG5cdGJhY2tncm91bmRSb3RhdGlvbj86IEV1bGVyO1xuXHRlbnZpcm9ubWVudEludGVuc2l0eT86IG51bWJlcjtcblx0ZW52aXJvbm1lbnRSb3RhdGlvbj86IEV1bGVyO1xuXG5cdG1hcD86IFRleHR1cmU7XG5cdHByZXNldD86IE5ndHNFbnZpcm9ubWVudFByZXNldHM7XG5cdHNjZW5lPzogU2NlbmUgfCBFbGVtZW50UmVmPFNjZW5lPjtcblx0Z3JvdW5kPzogYm9vbGVhbiB8IHsgcmFkaXVzPzogbnVtYmVyOyBoZWlnaHQ/OiBudW1iZXI7IHNjYWxlPzogbnVtYmVyIH07XG59XG5cbmNvbnN0IGRlZmF1bHRCYWNrZ3JvdW5kOiBOZ3RzRW52aXJvbm1lbnRPcHRpb25zID0ge1xuXHRiYWNrZ3JvdW5kOiBmYWxzZSxcbn07XG5cbkBEaXJlY3RpdmUoeyBzdGFuZGFsb25lOiB0cnVlLCBzZWxlY3RvcjogJ25ndHMtZW52aXJvbm1lbnQtbWFwJyB9KVxuZXhwb3J0IGNsYXNzIE5ndHNFbnZpcm9ubWVudE1hcCB7XG5cdG9wdGlvbnMgPSBpbnB1dChkZWZhdWx0QmFja2dyb3VuZCwgeyB0cmFuc2Zvcm06IG1lcmdlSW5wdXRzKGRlZmF1bHRCYWNrZ3JvdW5kKSB9KTtcblx0ZW52U2V0ID0gb3V0cHV0PHZvaWQ+KCk7XG5cblx0Y29uc3RydWN0b3IoKSB7XG5cdFx0Y29uc3Qgc3RvcmUgPSBpbmplY3RTdG9yZSgpO1xuXHRcdGNvbnN0IGRlZmF1bHRTY2VuZSA9IHN0b3JlLnNlbGVjdCgnc2NlbmUnKTtcblxuXHRcdGNvbnN0IF9tYXAgPSBwaWNrKHRoaXMub3B0aW9ucywgJ21hcCcpO1xuXHRcdGNvbnN0IF9lbnZDb25maWcgPSBjb21wdXRlZCgoKSA9PiB7XG5cdFx0XHRjb25zdCB7XG5cdFx0XHRcdGJhY2tncm91bmQgPSBmYWxzZSxcblx0XHRcdFx0c2NlbmUsXG5cdFx0XHRcdGJsdXIsXG5cdFx0XHRcdGJhY2tncm91bmRCbHVycmluZXNzLFxuXHRcdFx0XHRiYWNrZ3JvdW5kSW50ZW5zaXR5LFxuXHRcdFx0XHRiYWNrZ3JvdW5kUm90YXRpb24sXG5cdFx0XHRcdGVudmlyb25tZW50SW50ZW5zaXR5LFxuXHRcdFx0XHRlbnZpcm9ubWVudFJvdGF0aW9uLFxuXHRcdFx0fSA9IHRoaXMub3B0aW9ucygpO1xuXG5cdFx0XHRyZXR1cm4ge1xuXHRcdFx0XHRiYWNrZ3JvdW5kLFxuXHRcdFx0XHRzY2VuZSxcblx0XHRcdFx0Ymx1cixcblx0XHRcdFx0YmFja2dyb3VuZEJsdXJyaW5lc3MsXG5cdFx0XHRcdGJhY2tncm91bmRJbnRlbnNpdHksXG5cdFx0XHRcdGJhY2tncm91bmRSb3RhdGlvbixcblx0XHRcdFx0ZW52aXJvbm1lbnRJbnRlbnNpdHksXG5cdFx0XHRcdGVudmlyb25tZW50Um90YXRpb24sXG5cdFx0XHR9O1xuXHRcdH0pO1xuXG5cdFx0ZWZmZWN0KChvbkNsZWFudXApID0+IHtcblx0XHRcdGNvbnN0IG1hcCA9IF9tYXAoKTtcblx0XHRcdGlmICghbWFwKSByZXR1cm47XG5cdFx0XHRjb25zdCB7IGJhY2tncm91bmQgPSBmYWxzZSwgc2NlbmUsIC4uLmNvbmZpZyB9ID0gX2VudkNvbmZpZygpO1xuXHRcdFx0Y29uc3QgY2xlYW51cCA9IHNldEVudlByb3BzKGJhY2tncm91bmQsIHNjZW5lLCBkZWZhdWx0U2NlbmUoKSwgbWFwLCBjb25maWcpO1xuXHRcdFx0dGhpcy5lbnZTZXQuZW1pdCgpO1xuXHRcdFx0b25DbGVhbnVwKCgpID0+IGNsZWFudXAoKSk7XG5cdFx0fSk7XG5cdH1cbn1cblxuQERpcmVjdGl2ZSh7IHN0YW5kYWxvbmU6IHRydWUsIHNlbGVjdG9yOiAnbmd0cy1lbnZpcm9ubWVudC1jdWJlJyB9KVxuZXhwb3J0IGNsYXNzIE5ndHNFbnZpcm9ubWVudEN1YmUge1xuXHRvcHRpb25zID0gaW5wdXQoZGVmYXVsdEJhY2tncm91bmQsIHsgdHJhbnNmb3JtOiBtZXJnZUlucHV0cyhkZWZhdWx0QmFja2dyb3VuZCkgfSk7XG5cdGVudlNldCA9IG91dHB1dDx2b2lkPigpO1xuXG5cdHByaXZhdGUgc3RvcmUgPSBpbmplY3RTdG9yZSgpO1xuXHRwcml2YXRlIGRlZmF1bHRTY2VuZSA9IHRoaXMuc3RvcmUuc2VsZWN0KCdzY2VuZScpO1xuXG5cdHByaXZhdGUgZW52Q29uZmlnID0gY29tcHV0ZWQoKCkgPT4ge1xuXHRcdGNvbnN0IHtcblx0XHRcdGJhY2tncm91bmQgPSBmYWxzZSxcblx0XHRcdHNjZW5lLFxuXHRcdFx0Ymx1cixcblx0XHRcdGJhY2tncm91bmRCbHVycmluZXNzLFxuXHRcdFx0YmFja2dyb3VuZEludGVuc2l0eSxcblx0XHRcdGJhY2tncm91bmRSb3RhdGlvbixcblx0XHRcdGVudmlyb25tZW50SW50ZW5zaXR5LFxuXHRcdFx0ZW52aXJvbm1lbnRSb3RhdGlvbixcblx0XHR9ID0gdGhpcy5vcHRpb25zKCk7XG5cblx0XHRyZXR1cm4ge1xuXHRcdFx0YmFja2dyb3VuZCxcblx0XHRcdHNjZW5lLFxuXHRcdFx0Ymx1cixcblx0XHRcdGJhY2tncm91bmRCbHVycmluZXNzLFxuXHRcdFx0YmFja2dyb3VuZEludGVuc2l0eSxcblx0XHRcdGJhY2tncm91bmRSb3RhdGlvbixcblx0XHRcdGVudmlyb25tZW50SW50ZW5zaXR5LFxuXHRcdFx0ZW52aXJvbm1lbnRSb3RhdGlvbixcblx0XHR9O1xuXHR9KTtcblxuXHRjb25zdHJ1Y3RvcigpIHtcblx0XHRjb25zdCBfdGV4dHVyZSA9IGluamVjdEVudmlyb25tZW50KHRoaXMub3B0aW9ucyk7XG5cblx0XHRlZmZlY3QoKG9uQ2xlYW51cCkgPT4ge1xuXHRcdFx0Y29uc3QgdGV4dHVyZSA9IF90ZXh0dXJlKCk7XG5cdFx0XHRpZiAoIXRleHR1cmUpIHJldHVybjtcblx0XHRcdGNvbnN0IHsgYmFja2dyb3VuZCA9IGZhbHNlLCBzY2VuZSwgLi4uY29uZmlnIH0gPSB0aGlzLmVudkNvbmZpZygpO1xuXHRcdFx0Y29uc3QgY2xlYW51cCA9IHNldEVudlByb3BzKGJhY2tncm91bmQsIHNjZW5lLCB0aGlzLmRlZmF1bHRTY2VuZSgpLCB0ZXh0dXJlLCBjb25maWcpO1xuXHRcdFx0dGhpcy5lbnZTZXQuZW1pdCgpO1xuXHRcdFx0b25DbGVhbnVwKCgpID0+IGNsZWFudXAoKSk7XG5cdFx0fSk7XG5cdH1cbn1cblxuQENvbXBvbmVudCh7XG5cdHNlbGVjdG9yOiAnbmd0cy1lbnZpcm9ubWVudC1wb3J0YWwnLFxuXHRzdGFuZGFsb25lOiB0cnVlLFxuXHR0ZW1wbGF0ZTogYFxuXHRcdDxuZ3QtcG9ydGFsIFtjb250YWluZXJdPVwidmlydHVhbFNjZW5lXCI+XG5cdFx0XHQ8bmctdGVtcGxhdGUgcG9ydGFsQ29udGVudCBsZXQtaW5qZWN0b3I9XCJpbmplY3RvclwiIGxldC1jb250YWluZXI9XCJjb250YWluZXJcIj5cblx0XHRcdFx0PG5nLWNvbnRhaW5lclxuXHRcdFx0XHRcdFtuZ1RlbXBsYXRlT3V0bGV0XT1cImNvbnRlbnQoKVwiXG5cdFx0XHRcdFx0W25nVGVtcGxhdGVPdXRsZXRJbmplY3Rvcl09XCJpbmplY3RvclwiXG5cdFx0XHRcdFx0W25nVGVtcGxhdGVPdXRsZXRDb250ZXh0XT1cInsgaW5qZWN0b3IsIGNvbnRhaW5lciB9XCJcblx0XHRcdFx0Lz5cblxuXHRcdFx0XHQ8bmd0LWN1YmUtY2FtZXJhICNjdWJlQ2FtZXJhICphcmdzPVwiY2FtZXJhQXJncygpXCIgLz5cblxuXHRcdFx0XHRAaWYgKGZpbGVzKCkgfHwgcHJlc2V0KCkpIHtcblx0XHRcdFx0XHQ8bmd0cy1lbnZpcm9ubWVudC1jdWJlIFtvcHRpb25zXT1cImVudkN1YmVPcHRpb25zKClcIiAoZW52U2V0KT1cIm9uRW52U2V0KClcIiAvPlxuXHRcdFx0XHR9IEBlbHNlIGlmIChtYXAoKSkge1xuXHRcdFx0XHRcdDxuZ3RzLWVudmlyb25tZW50LW1hcCBbb3B0aW9uc109XCJlbnZNYXBPcHRpb25zKClcIiAoZW52U2V0KT1cIm9uRW52U2V0KClcIiAvPlxuXHRcdFx0XHR9XG5cdFx0XHQ8L25nLXRlbXBsYXRlPlxuXHRcdDwvbmd0LXBvcnRhbD5cblx0YCxcblx0aW1wb3J0czogW05ndHNFbnZpcm9ubWVudEN1YmUsIE5ndHNFbnZpcm9ubWVudE1hcCwgTmd0QXJncywgTmd0UG9ydGFsLCBOZ3RQb3J0YWxDb250ZW50LCBOZ1RlbXBsYXRlT3V0bGV0XSxcblx0c2NoZW1hczogW0NVU1RPTV9FTEVNRU5UU19TQ0hFTUFdLFxuXHRjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgTmd0c0Vudmlyb25tZW50UG9ydGFsIHtcblx0cHJpdmF0ZSBkZWZhdWx0T3B0aW9uczogTmd0c0Vudmlyb25tZW50T3B0aW9ucyA9IHtcblx0XHRuZWFyOiAxLFxuXHRcdGZhcjogMTAwMCxcblx0XHRyZXNvbHV0aW9uOiAyNTYsXG5cdFx0ZnJhbWVzOiAxLFxuXHRcdGJhY2tncm91bmQ6IGZhbHNlLFxuXHR9O1xuXHRvcHRpb25zID0gaW5wdXQodGhpcy5kZWZhdWx0T3B0aW9ucywgeyB0cmFuc2Zvcm06IG1lcmdlSW5wdXRzKHRoaXMuZGVmYXVsdE9wdGlvbnMpIH0pO1xuXHRjb250ZW50ID0gaW5wdXQucmVxdWlyZWQ8VGVtcGxhdGVSZWY8dW5rbm93bj4+KCk7XG5cdGVudlNldCA9IG91dHB1dDx2b2lkPigpO1xuXG5cdHByaXZhdGUgaW5qZWN0b3IgPSBpbmplY3QoSW5qZWN0b3IpO1xuXHRwcml2YXRlIHN0b3JlID0gaW5qZWN0U3RvcmUoKTtcblx0cHJpdmF0ZSBkZWZhdWx0U2NlbmUgPSB0aGlzLnN0b3JlLnNlbGVjdCgnc2NlbmUnKTtcblx0cHJpdmF0ZSBnbCA9IHRoaXMuc3RvcmUuc2VsZWN0KCdnbCcpO1xuXG5cdHByaXZhdGUgY2FtZXJhUmVmID0gdmlld0NoaWxkPEVsZW1lbnRSZWY8Q3ViZUNhbWVyYT4+KCdjdWJlQ2FtZXJhJyk7XG5cblx0cHJvdGVjdGVkIG1hcCA9IHBpY2sodGhpcy5vcHRpb25zLCAnbWFwJyk7XG5cdHByb3RlY3RlZCBmaWxlcyA9IHBpY2sodGhpcy5vcHRpb25zLCAnZmlsZXMnKTtcblx0cHJvdGVjdGVkIHByZXNldCA9IHBpY2sodGhpcy5vcHRpb25zLCAncHJlc2V0Jyk7XG5cdHByaXZhdGUgZXh0ZW5zaW9ucyA9IHBpY2sodGhpcy5vcHRpb25zLCAnZXh0ZW5zaW9ucycpO1xuXHRwcml2YXRlIHBhdGggPSBwaWNrKHRoaXMub3B0aW9ucywgJ3BhdGgnKTtcblxuXHRwcm90ZWN0ZWQgZW52TWFwT3B0aW9ucyA9IGNvbXB1dGVkKCgpID0+ICh7IGJhY2tncm91bmQ6IHRydWUsIG1hcDogdGhpcy5tYXAoKSwgZXh0ZW5zaW9uczogdGhpcy5leHRlbnNpb25zKCkgfSkpO1xuXHRwcm90ZWN0ZWQgZW52Q3ViZU9wdGlvbnMgPSBjb21wdXRlZCgoKSA9PiAoe1xuXHRcdGJhY2tncm91bmQ6IHRydWUsXG5cdFx0ZmlsZXM6IHRoaXMuZmlsZXMoKSxcblx0XHRwcmVzZXQ6IHRoaXMucHJlc2V0KCksXG5cdFx0ZXh0ZW5zaW9uczogdGhpcy5leHRlbnNpb25zKCksXG5cdFx0cGF0aDogdGhpcy5wYXRoKCksXG5cdH0pKTtcblxuXHRwcml2YXRlIG5lYXIgPSBwaWNrKHRoaXMub3B0aW9ucywgJ25lYXInKTtcblx0cHJpdmF0ZSBmYXIgPSBwaWNrKHRoaXMub3B0aW9ucywgJ2ZhcicpO1xuXHRwcml2YXRlIHJlc29sdXRpb24gPSBwaWNrKHRoaXMub3B0aW9ucywgJ3Jlc29sdXRpb24nKTtcblx0cHJpdmF0ZSBmYm8gPSBjb21wdXRlZCgoKSA9PiB7XG5cdFx0Y29uc3QgZmJvID0gbmV3IFdlYkdMQ3ViZVJlbmRlclRhcmdldCh0aGlzLnJlc29sdXRpb24oKSk7XG5cdFx0ZmJvLnRleHR1cmUudHlwZSA9IEhhbGZGbG9hdFR5cGU7XG5cdFx0cmV0dXJuIGZibztcblx0fSk7XG5cblx0cHJvdGVjdGVkIGNhbWVyYUFyZ3MgPSBjb21wdXRlZCgoKSA9PiBbdGhpcy5uZWFyKCksIHRoaXMuZmFyKCksIHRoaXMuZmJvKCldKTtcblx0cHJvdGVjdGVkIHZpcnR1YWxTY2VuZSA9IHByZXBhcmUobmV3IFNjZW5lKCkpO1xuXG5cdHByaXZhdGUgc2V0RW52RWZmZWN0UmVmPzogRWZmZWN0UmVmO1xuXG5cdGNvbnN0cnVjdG9yKCkge1xuXHRcdGV4dGVuZCh7IEN1YmVDYW1lcmEgfSk7XG5cblx0XHRlZmZlY3QoKG9uQ2xlYW51cCkgPT4ge1xuXHRcdFx0Y29uc3QgW2ZpbGVzLCBwcmVzZXQsIG1hcF0gPSBbdGhpcy5maWxlcygpLCB0aGlzLnByZXNldCgpLCB0aGlzLm1hcCgpXTtcblx0XHRcdC8vIE5PVEU6IHdoZW4gdGhlcmUncyBub25lIG9mIHRoaXMsIHdlIGRvbid0IHJlbmRlciBjdWJlIG9yIG1hcCBzbyB3ZSBuZWVkIHRvIHNldEVudiBoZXJlXG5cdFx0XHRpZiAoISFmaWxlcyB8fCAhIXByZXNldCB8fCAhIW1hcCkgcmV0dXJuO1xuXHRcdFx0Y29uc3QgY2xlYW51cCA9IHRoaXMuc2V0UG9ydGFsRW52KCk7XG5cdFx0XHRvbkNsZWFudXAoKCkgPT4gY2xlYW51cD8uKCkpO1xuXHRcdH0pO1xuXG5cdFx0bGV0IGNvdW50ID0gMTtcblx0XHRpbmplY3RCZWZvcmVSZW5kZXIoKCkgPT4ge1xuXHRcdFx0Y29uc3QgZnJhbWVzID0gdGhpcy5vcHRpb25zKCkuZnJhbWVzO1xuXHRcdFx0aWYgKGZyYW1lcyA9PT0gSW5maW5pdHkgfHwgKGZyYW1lcyAhPSBudWxsICYmIGNvdW50IDwgZnJhbWVzKSkge1xuXHRcdFx0XHRjb25zdCBjYW1lcmEgPSB0aGlzLmNhbWVyYVJlZigpPy5uYXRpdmVFbGVtZW50O1xuXHRcdFx0XHRpZiAoY2FtZXJhKSB7XG5cdFx0XHRcdFx0Y2FtZXJhLnVwZGF0ZSh0aGlzLmdsKCksIHRoaXMudmlydHVhbFNjZW5lKTtcblx0XHRcdFx0XHRjb3VudCsrO1xuXHRcdFx0XHR9XG5cdFx0XHR9XG5cdFx0fSk7XG5cblx0XHRpbmplY3QoRGVzdHJveVJlZikub25EZXN0cm95KCgpID0+IHtcblx0XHRcdGlmICh0aGlzLnNldEVudkVmZmVjdFJlZikgdGhpcy5zZXRFbnZFZmZlY3RSZWYuZGVzdHJveSgpO1xuXHRcdH0pO1xuXHR9XG5cblx0Ly8gTk9URTogd2UgdXNlIG9uRW52U2V0IGhlcmUgdG8gZW5zdXJlIHRoYXQgRW52aXJvbm1lbnRDdWJlIG9yIEVudmlyb25tZW50TWFwIHNldHMgdGhlIGVudiBiZWZvcmUgdGhlIHBvcnRhbFxuXHRvbkVudlNldCgpIHtcblx0XHRpZiAodGhpcy5zZXRFbnZFZmZlY3RSZWYpIHRoaXMuc2V0RW52RWZmZWN0UmVmLmRlc3Ryb3koKTtcblx0XHR0aGlzLnNldEVudkVmZmVjdFJlZiA9IGVmZmVjdChcblx0XHRcdChvbkNsZWFudXApID0+IHtcblx0XHRcdFx0Y29uc3QgY2xlYW51cCA9IHRoaXMuc2V0UG9ydGFsRW52KCk7XG5cdFx0XHRcdG9uQ2xlYW51cCgoKSA9PiBjbGVhbnVwPy4oKSk7XG5cdFx0XHR9LFxuXHRcdFx0eyBtYW51YWxDbGVhbnVwOiB0cnVlLCBpbmplY3RvcjogdGhpcy5pbmplY3RvciB9LFxuXHRcdCk7XG5cdH1cblxuXHRwcml2YXRlIHNldFBvcnRhbEVudigpIHtcblx0XHRjb25zdCBjYW1lcmEgPSB0aGlzLmNhbWVyYVJlZigpO1xuXHRcdGlmICghY2FtZXJhPy5uYXRpdmVFbGVtZW50KSByZXR1cm47XG5cblx0XHRjb25zdCBbXG5cdFx0XHR7XG5cdFx0XHRcdGZyYW1lcyxcblx0XHRcdFx0YmFja2dyb3VuZCA9IGZhbHNlLFxuXHRcdFx0XHRzY2VuZSxcblx0XHRcdFx0Ymx1cixcblx0XHRcdFx0YmFja2dyb3VuZEJsdXJyaW5lc3MsXG5cdFx0XHRcdGJhY2tncm91bmRJbnRlbnNpdHksXG5cdFx0XHRcdGJhY2tncm91bmRSb3RhdGlvbixcblx0XHRcdFx0ZW52aXJvbm1lbnRJbnRlbnNpdHksXG5cdFx0XHRcdGVudmlyb25tZW50Um90YXRpb24sXG5cdFx0XHR9LFxuXHRcdFx0Z2wsXG5cdFx0XHRmYm8sXG5cdFx0XHRkZWZhdWx0U2NlbmUsXG5cdFx0XSA9IFt0aGlzLm9wdGlvbnMoKSwgdGhpcy5nbCgpLCB0aGlzLmZibygpLCB0aGlzLmRlZmF1bHRTY2VuZSgpXTtcblxuXHRcdGlmIChmcmFtZXMgPT09IDEpIGNhbWVyYS5uYXRpdmVFbGVtZW50LnVwZGF0ZShnbCwgdGhpcy52aXJ0dWFsU2NlbmUpO1xuXHRcdGNvbnN0IGNsZWFudXAgPSBzZXRFbnZQcm9wcyhiYWNrZ3JvdW5kLCBzY2VuZSwgZGVmYXVsdFNjZW5lLCBmYm8udGV4dHVyZSwge1xuXHRcdFx0Ymx1cixcblx0XHRcdGJhY2tncm91bmRCbHVycmluZXNzLFxuXHRcdFx0YmFja2dyb3VuZEludGVuc2l0eSxcblx0XHRcdGJhY2tncm91bmRSb3RhdGlvbixcblx0XHRcdGVudmlyb25tZW50SW50ZW5zaXR5LFxuXHRcdFx0ZW52aXJvbm1lbnRSb3RhdGlvbixcblx0XHR9KTtcblx0XHR0aGlzLmVudlNldC5lbWl0KCk7XG5cdFx0cmV0dXJuIGNsZWFudXA7XG5cdH1cbn1cblxuQENvbXBvbmVudCh7XG5cdHNlbGVjdG9yOiAnbmd0cy1lbnZpcm9ubWVudC1ncm91bmQnLFxuXHRzdGFuZGFsb25lOiB0cnVlLFxuXHR0ZW1wbGF0ZTogYFxuXHRcdDxuZ3RzLWVudmlyb25tZW50LW1hcCBbb3B0aW9uc109XCJlbnZNYXBPcHRpb25zKClcIiAoZW52U2V0KT1cImVudlNldC5lbWl0KClcIiAvPlxuXHRcdDxuZ3QtZ3JvdW5kLXByb2plY3RlZC1lbnYgKmFyZ3M9XCJhcmdzKClcIiBbc2NhbGVdPVwic2NhbGUoKVwiIFtoZWlnaHRdPVwiaGVpZ2h0KClcIiBbcmFkaXVzXT1cInJhZGl1cygpXCIgLz5cblx0YCxcblx0aW1wb3J0czogW05ndHNFbnZpcm9ubWVudE1hcCwgTmd0QXJnc10sXG5cdHNjaGVtYXM6IFtDVVNUT01fRUxFTUVOVFNfU0NIRU1BXSxcblx0Y2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG59KVxuZXhwb3J0IGNsYXNzIE5ndHNFbnZpcm9ubWVudEdyb3VuZCB7XG5cdG9wdGlvbnMgPSBpbnB1dCh7fSBhcyBOZ3RzRW52aXJvbm1lbnRPcHRpb25zKTtcblx0ZW52U2V0ID0gb3V0cHV0PHZvaWQ+KCk7XG5cblx0cHJpdmF0ZSBkZWZhdWx0VGV4dHVyZSA9IGluamVjdEVudmlyb25tZW50KHRoaXMub3B0aW9ucyk7XG5cblx0cHJvdGVjdGVkIGhlaWdodCA9IGNvbXB1dGVkKCgpID0+ICh0aGlzLm9wdGlvbnMoKS5ncm91bmQgYXMgYW55KT8uaGVpZ2h0KTtcblx0cHJvdGVjdGVkIHJhZGl1cyA9IGNvbXB1dGVkKCgpID0+ICh0aGlzLm9wdGlvbnMoKS5ncm91bmQgYXMgYW55KT8ucmFkaXVzKTtcblx0cHJvdGVjdGVkIHNjYWxlID0gY29tcHV0ZWQoKCkgPT4gKHRoaXMub3B0aW9ucygpLmdyb3VuZCBhcyBhbnkpPy5zY2FsZSA/PyAxMDAwKTtcblx0cHJvdGVjdGVkIGFyZ3MgPSBjb21wdXRlZCgoKSA9PiBbdGhpcy5vcHRpb25zKCkubWFwIHx8IHRoaXMuZGVmYXVsdFRleHR1cmUoKV0pO1xuXHRwcm90ZWN0ZWQgZW52TWFwT3B0aW9ucyA9IGNvbXB1dGVkKCgpID0+IHtcblx0XHRjb25zdCB7IG1hcDogXywgLi4ub3B0aW9ucyB9ID0gdGhpcy5vcHRpb25zKCk7XG5cdFx0Y29uc3QgW21hcF0gPSB0aGlzLmFyZ3MoKTtcblx0XHRyZXR1cm4gT2JqZWN0LmFzc2lnbihvcHRpb25zLCB7IG1hcCB9KSBhcyBOZ3RzRW52aXJvbm1lbnRPcHRpb25zO1xuXHR9KTtcblxuXHRjb25zdHJ1Y3RvcigpIHtcblx0XHRleHRlbmQoeyBHcm91bmRQcm9qZWN0ZWRFbnYgfSk7XG5cdH1cbn1cblxuQENvbXBvbmVudCh7XG5cdHNlbGVjdG9yOiAnbmd0cy1lbnZpcm9ubWVudCcsXG5cdHN0YW5kYWxvbmU6IHRydWUsXG5cdHRlbXBsYXRlOiBgXG5cdFx0QGlmIChvcHRpb25zKCkuZ3JvdW5kKSB7XG5cdFx0XHQ8bmd0cy1lbnZpcm9ubWVudC1ncm91bmQgW29wdGlvbnNdPVwib3B0aW9ucygpXCIgKGVudlNldCk9XCJlbnZTZXQuZW1pdCgpXCIgLz5cblx0XHR9IEBlbHNlIGlmIChvcHRpb25zKCkubWFwKSB7XG5cdFx0XHQ8bmd0cy1lbnZpcm9ubWVudC1tYXAgW29wdGlvbnNdPVwib3B0aW9ucygpXCIgKGVudlNldCk9XCJlbnZTZXQuZW1pdCgpXCIgLz5cblx0XHR9IEBlbHNlIGlmIChjb250ZW50KCkpIHtcblx0XHRcdDxuZ3RzLWVudmlyb25tZW50LXBvcnRhbCBbb3B0aW9uc109XCJvcHRpb25zKClcIiBbY29udGVudF09XCIkYW55KGNvbnRlbnQoKSlcIiAoZW52U2V0KT1cImVudlNldC5lbWl0KClcIiAvPlxuXHRcdH0gQGVsc2Uge1xuXHRcdFx0PG5ndHMtZW52aXJvbm1lbnQtY3ViZSBbb3B0aW9uc109XCJvcHRpb25zKClcIiAoZW52U2V0KT1cImVudlNldC5lbWl0KClcIiAvPlxuXHRcdH1cblx0YCxcblx0aW1wb3J0czogW05ndHNFbnZpcm9ubWVudEN1YmUsIE5ndHNFbnZpcm9ubWVudE1hcCwgTmd0c0Vudmlyb25tZW50UG9ydGFsLCBOZ3RzRW52aXJvbm1lbnRHcm91bmRdLFxuXHRzY2hlbWFzOiBbQ1VTVE9NX0VMRU1FTlRTX1NDSEVNQV0sXG5cdGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxufSlcbmV4cG9ydCBjbGFzcyBOZ3RzRW52aXJvbm1lbnQge1xuXHRvcHRpb25zID0gaW5wdXQoe30gYXMgTmd0c0Vudmlyb25tZW50T3B0aW9ucyk7XG5cdGNvbnRlbnQgPSBjb250ZW50Q2hpbGQoVGVtcGxhdGVSZWYpO1xuXHRlbnZTZXQgPSBvdXRwdXQ8dm9pZD4oKTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,184 @@
|
|
|
1
|
+
import { computed, effect, signal, untracked } from '@angular/core';
|
|
2
|
+
import { GainMapLoader, HDRJPGLoader } from '@monogrid/gainmap-js';
|
|
3
|
+
import { injectLoader, injectStore, pick } from 'angular-three';
|
|
4
|
+
import { LinearEncoding, sRGBEncoding } from 'angular-three-soba/misc';
|
|
5
|
+
import { assertInjector } from 'ngxtension/assert-injector';
|
|
6
|
+
import { CubeReflectionMapping, CubeTexture, CubeTextureLoader, EquirectangularReflectionMapping, } from 'three';
|
|
7
|
+
import { EXRLoader, RGBELoader } from 'three-stdlib';
|
|
8
|
+
export const ENVIRONMENT_PRESETS = {
|
|
9
|
+
apartment: 'lebombo_1k.hdr',
|
|
10
|
+
city: 'potsdamer_platz_1k.hdr',
|
|
11
|
+
dawn: 'kiara_1_dawn_1k.hdr',
|
|
12
|
+
forest: 'forest_slope_1k.hdr',
|
|
13
|
+
lobby: 'st_fagans_interior_1k.hdr',
|
|
14
|
+
night: 'dikhololo_night_1k.hdr',
|
|
15
|
+
park: 'rooitou_park_1k.hdr',
|
|
16
|
+
studio: 'studio_small_03_1k.hdr',
|
|
17
|
+
sunset: 'venice_sunset_1k.hdr',
|
|
18
|
+
warehouse: 'empty_warehouse_01_1k.hdr',
|
|
19
|
+
};
|
|
20
|
+
const CUBEMAP_ROOT = 'https://raw.githack.com/pmndrs/drei-assets/456060a26bbeb8fdf79326f224b6d99b8bcce736/hdri/';
|
|
21
|
+
const defaultFiles = ['/px.png', '/nx.png', '/py.png', '/ny.png', '/pz.png', '/nz.png'];
|
|
22
|
+
export function injectEnvironment(options = () => ({}), { injector } = {}) {
|
|
23
|
+
return assertInjector(injectEnvironment, injector, () => {
|
|
24
|
+
const adjustedOptions = computed(() => {
|
|
25
|
+
const { preset, extensions, encoding, ...rest } = options();
|
|
26
|
+
let { files, path } = rest;
|
|
27
|
+
if (files == null) {
|
|
28
|
+
files = defaultFiles;
|
|
29
|
+
}
|
|
30
|
+
if (path == null) {
|
|
31
|
+
path = '';
|
|
32
|
+
}
|
|
33
|
+
if (preset) {
|
|
34
|
+
validatePreset(preset);
|
|
35
|
+
files = ENVIRONMENT_PRESETS[preset];
|
|
36
|
+
path = CUBEMAP_ROOT;
|
|
37
|
+
}
|
|
38
|
+
return { files, preset, encoding, path, extensions };
|
|
39
|
+
});
|
|
40
|
+
const files = pick(adjustedOptions, 'files');
|
|
41
|
+
const multiFile = computed(() => Array.isArray(files()));
|
|
42
|
+
const resultOptions = computed(() => getExtension(files()));
|
|
43
|
+
const extension = pick(resultOptions, 'extension');
|
|
44
|
+
const loader = computed(() => getLoader(extension()));
|
|
45
|
+
const store = injectStore();
|
|
46
|
+
const gl = store.select('gl');
|
|
47
|
+
const texture = signal(null);
|
|
48
|
+
effect(() => {
|
|
49
|
+
const [_extension, _multiFile, _files] = [untracked(extension), untracked(multiFile), files()];
|
|
50
|
+
if (_extension !== 'webp' && _extension !== 'jpg' && _extension !== 'jpeg')
|
|
51
|
+
return;
|
|
52
|
+
gl().domElement.addEventListener('webglcontextlost', () => {
|
|
53
|
+
// @ts-expect-error - files is correctly passed
|
|
54
|
+
injectLoader.clear(multiFile ? [_files] : _files);
|
|
55
|
+
}, { once: true });
|
|
56
|
+
});
|
|
57
|
+
const result = injectLoader(loader,
|
|
58
|
+
// @ts-expect-error - ensure the files is an array
|
|
59
|
+
() => {
|
|
60
|
+
const { files } = adjustedOptions();
|
|
61
|
+
return Array.isArray(files) ? [files] : files;
|
|
62
|
+
}, {
|
|
63
|
+
extensions: (loader) => {
|
|
64
|
+
const { extensions, path } = adjustedOptions();
|
|
65
|
+
const { extension } = resultOptions();
|
|
66
|
+
if (extension === 'webp' || extension === 'jpg' || extension === 'jpeg') {
|
|
67
|
+
// @ts-expect-error - Gainmap requires a renderer
|
|
68
|
+
loader.setRenderer(gl());
|
|
69
|
+
}
|
|
70
|
+
loader.setPath?.(path);
|
|
71
|
+
if (extensions)
|
|
72
|
+
extensions(loader);
|
|
73
|
+
},
|
|
74
|
+
});
|
|
75
|
+
effect(() => {
|
|
76
|
+
const loaderResult = result();
|
|
77
|
+
if (!loaderResult)
|
|
78
|
+
return;
|
|
79
|
+
untracked(() => {
|
|
80
|
+
const { extension, isCubeMap } = resultOptions();
|
|
81
|
+
const _multiFile = multiFile();
|
|
82
|
+
const { encoding } = adjustedOptions();
|
|
83
|
+
// @ts-expect-error - ensure textureResult is a Texture or CubeTexture
|
|
84
|
+
let textureResult = (_multiFile ? loaderResult[0] : loaderResult);
|
|
85
|
+
// NOTE: racing condition, we can skip this
|
|
86
|
+
// we just said above that if multiFile is false, it is a single Texture
|
|
87
|
+
if (!_multiFile && Array.isArray(textureResult) && textureResult[0] instanceof CubeTexture) {
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
if (!(textureResult instanceof CubeTexture) &&
|
|
91
|
+
(extension === 'jpg' || extension === 'jpeg' || extension === 'webp')) {
|
|
92
|
+
textureResult = textureResult.renderTarget?.texture;
|
|
93
|
+
}
|
|
94
|
+
textureResult.mapping = isCubeMap ? CubeReflectionMapping : EquirectangularReflectionMapping;
|
|
95
|
+
if ('colorSpace' in textureResult)
|
|
96
|
+
textureResult.colorSpace = encoding ?? (isCubeMap ? 'srgb' : 'srgb-linear');
|
|
97
|
+
else
|
|
98
|
+
textureResult.encoding = encoding ?? (isCubeMap ? sRGBEncoding : LinearEncoding);
|
|
99
|
+
texture.set(textureResult);
|
|
100
|
+
});
|
|
101
|
+
});
|
|
102
|
+
return texture.asReadonly();
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
injectEnvironment.preload = (options = () => ({})) => {
|
|
106
|
+
const _options = options();
|
|
107
|
+
let { files, path } = _options;
|
|
108
|
+
const { preset, extensions } = _options;
|
|
109
|
+
if (files == null) {
|
|
110
|
+
files = defaultFiles;
|
|
111
|
+
}
|
|
112
|
+
if (path == null) {
|
|
113
|
+
path = '';
|
|
114
|
+
}
|
|
115
|
+
if (preset) {
|
|
116
|
+
validatePreset(preset);
|
|
117
|
+
files = ENVIRONMENT_PRESETS[preset];
|
|
118
|
+
path = CUBEMAP_ROOT;
|
|
119
|
+
}
|
|
120
|
+
const { extension } = getExtension(files);
|
|
121
|
+
if (extension === 'webp' || extension === 'jpg' || extension === 'jpeg') {
|
|
122
|
+
throw new Error('injectEnvironment: Preloading gainmaps is not supported');
|
|
123
|
+
}
|
|
124
|
+
const loader = getLoader(extension);
|
|
125
|
+
if (!loader)
|
|
126
|
+
throw new Error('injectEnvironment: Unrecognized file extension: ' + files);
|
|
127
|
+
injectLoader.preload(() => loader,
|
|
128
|
+
// @ts-expect-error - files is correctly passed
|
|
129
|
+
() => (Array.isArray(files) ? [files] : files), (loader) => {
|
|
130
|
+
loader.setPath?.(path);
|
|
131
|
+
if (extensions)
|
|
132
|
+
extensions(loader);
|
|
133
|
+
});
|
|
134
|
+
};
|
|
135
|
+
injectEnvironment.clear = (clearOptions) => {
|
|
136
|
+
const options = { files: defaultFiles, ...clearOptions };
|
|
137
|
+
let { files } = options;
|
|
138
|
+
const preset = options.preset;
|
|
139
|
+
if (preset) {
|
|
140
|
+
validatePreset(preset);
|
|
141
|
+
files = ENVIRONMENT_PRESETS[preset];
|
|
142
|
+
}
|
|
143
|
+
injectLoader.clear(files);
|
|
144
|
+
};
|
|
145
|
+
function validatePreset(preset) {
|
|
146
|
+
if (!(preset in ENVIRONMENT_PRESETS))
|
|
147
|
+
throw new Error('Preset must be one of: ' + Object.keys(ENVIRONMENT_PRESETS).join(', '));
|
|
148
|
+
}
|
|
149
|
+
function getExtension(files) {
|
|
150
|
+
const isCubeMap = Array.isArray(files) && files.length === 6;
|
|
151
|
+
const isGainmap = Array.isArray(files) && files.length === 3 && files.some((file) => file.endsWith('json'));
|
|
152
|
+
const firstEntry = Array.isArray(files) ? files[0] : files;
|
|
153
|
+
// Everything else
|
|
154
|
+
const extension = isCubeMap
|
|
155
|
+
? 'cube'
|
|
156
|
+
: isGainmap
|
|
157
|
+
? 'webp'
|
|
158
|
+
: firstEntry.startsWith('data:application/exr')
|
|
159
|
+
? 'exr'
|
|
160
|
+
: firstEntry.startsWith('data:application/hdr')
|
|
161
|
+
? 'hdr'
|
|
162
|
+
: firstEntry.startsWith('data:image/jpeg')
|
|
163
|
+
? 'jpg'
|
|
164
|
+
: firstEntry.split('.').pop()?.split('?')?.shift()?.toLowerCase();
|
|
165
|
+
return { extension, isCubeMap, isGainmap };
|
|
166
|
+
}
|
|
167
|
+
function getLoader(extension) {
|
|
168
|
+
const loader = extension === 'cube'
|
|
169
|
+
? CubeTextureLoader
|
|
170
|
+
: extension === 'hdr'
|
|
171
|
+
? RGBELoader
|
|
172
|
+
: extension === 'exr'
|
|
173
|
+
? EXRLoader
|
|
174
|
+
: extension === 'jpg' || extension === 'jpeg'
|
|
175
|
+
? HDRJPGLoader
|
|
176
|
+
: extension === 'webp'
|
|
177
|
+
? GainMapLoader
|
|
178
|
+
: null;
|
|
179
|
+
if (!loader) {
|
|
180
|
+
throw new Error('injectEnvironment: Unrecognized file extension: ' + extension);
|
|
181
|
+
}
|
|
182
|
+
return loader;
|
|
183
|
+
}
|
|
184
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5qZWN0LWVudmlyb25tZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9zb2JhL3N0YWdpbmcvc3JjL2xpYi9lbnZpcm9ubWVudC9pbmplY3QtZW52aXJvbm1lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQVksTUFBTSxFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUM5RSxPQUFPLEVBQUUsYUFBYSxFQUFFLFlBQVksRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBQ25FLE9BQU8sRUFBRSxZQUFZLEVBQUUsV0FBVyxFQUFFLElBQUksRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNoRSxPQUFPLEVBQUUsY0FBYyxFQUFFLFlBQVksRUFBbUIsTUFBTSx5QkFBeUIsQ0FBQztBQUN4RixPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDNUQsT0FBTyxFQUNOLHFCQUFxQixFQUNyQixXQUFXLEVBQ1gsaUJBQWlCLEVBQ2pCLGdDQUFnQyxHQUdoQyxNQUFNLE9BQU8sQ0FBQztBQUNmLE9BQU8sRUFBRSxTQUFTLEVBQUUsVUFBVSxFQUFFLE1BQU0sY0FBYyxDQUFDO0FBRXJELE1BQU0sQ0FBQyxNQUFNLG1CQUFtQixHQUFHO0lBQ2xDLFNBQVMsRUFBRSxnQkFBZ0I7SUFDM0IsSUFBSSxFQUFFLHdCQUF3QjtJQUM5QixJQUFJLEVBQUUscUJBQXFCO0lBQzNCLE1BQU0sRUFBRSxxQkFBcUI7SUFDN0IsS0FBSyxFQUFFLDJCQUEyQjtJQUNsQyxLQUFLLEVBQUUsd0JBQXdCO0lBQy9CLElBQUksRUFBRSxxQkFBcUI7SUFDM0IsTUFBTSxFQUFFLHdCQUF3QjtJQUNoQyxNQUFNLEVBQUUsc0JBQXNCO0lBQzlCLFNBQVMsRUFBRSwyQkFBMkI7Q0FDdEMsQ0FBQztBQUlGLE1BQU0sWUFBWSxHQUFHLDJGQUEyRixDQUFDO0FBVWpI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LFNBQVMsS0FBSyxLQUFLLElBQUksU0FBUyxLQUFLLE1BQU07b0JBQzVDLENBQUMsQ0FBRSxZQUF5QztvQkFDNUMsQ0FBQyxDQUFDLFNBQVMsS0FBSyxNQUFNO3dCQUNyQixDQUFDLENBQUUsYUFBMEM7d0JBQzdDLENBQUMsQ0FBQyxJQUFJLENBQUM7SUFFYixJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDYixNQUFNLElBQUksS0FBSyxDQUFDLGtEQUFrRCxHQUFHLFNBQVMsQ0FBQyxDQUFDO0lBQ2pGLENBQUM7SUFFRCxPQUFPLE1BQXVCLENBQUM7QUFDaEMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNvbXB1dGVkLCBlZmZlY3QsIEluamVjdG9yLCBzaWduYWwsIHVudHJhY2tlZCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgR2Fpbk1hcExvYWRlciwgSERSSlBHTG9hZGVyIH0gZnJvbSAnQG1vbm9ncmlkL2dhaW5tYXAtanMnO1xuaW1wb3J0IHsgaW5qZWN0TG9hZGVyLCBpbmplY3RTdG9yZSwgcGljayB9IGZyb20gJ2FuZ3VsYXItdGhyZWUnO1xuaW1wb3J0IHsgTGluZWFyRW5jb2RpbmcsIHNSR0JFbmNvZGluZywgVGV4dHVyZUVuY29kaW5nIH0gZnJvbSAnYW5ndWxhci10aHJlZS1zb2JhL21pc2MnO1xuaW1wb3J0IHsgYXNzZXJ0SW5qZWN0b3IgfSBmcm9tICduZ3h0ZW5zaW9uL2Fzc2VydC1pbmplY3Rvcic7XG5pbXBvcnQge1xuXHRDdWJlUmVmbGVjdGlvbk1hcHBpbmcsXG5cdEN1YmVUZXh0dXJlLFxuXHRDdWJlVGV4dHVyZUxvYWRlcixcblx0RXF1aXJlY3Rhbmd1bGFyUmVmbGVjdGlvbk1hcHBpbmcsXG5cdExvYWRlcixcblx0VGV4dHVyZSxcbn0gZnJvbSAndGhyZWUnO1xuaW1wb3J0IHsgRVhSTG9hZGVyLCBSR0JFTG9hZGVyIH0gZnJvbSAndGhyZWUtc3RkbGliJztcblxuZXhwb3J0IGNvbnN0IEVOVklST05NRU5UX1BSRVNFVFMgPSB7XG5cdGFwYXJ0bWVudDogJ2xlYm9tYm9fMWsuaGRyJyxcblx0Y2l0eTogJ3BvdHNkYW1lcl9wbGF0el8xay5oZHInLFxuXHRkYXduOiAna2lhcmFfMV9kYXduXzFrLmhkcicsXG5cdGZvcmVzdDogJ2ZvcmVzdF9zbG9wZV8xay5oZHInLFxuXHRsb2JieTogJ3N0X2ZhZ2Fuc19pbnRlcmlvcl8xay5oZHInLFxuXHRuaWdodDogJ2Rpa2hvbG9sb19uaWdodF8xay5oZHInLFxuXHRwYXJrOiAncm9vaXRvdV9wYXJrXzFrLmhkcicsXG5cdHN0dWRpbzogJ3N0dWRpb19zbWFsbF8wM18xay5oZHInLFxuXHRzdW5zZXQ6ICd2ZW5pY2Vfc3Vuc2V0XzFrLmhkcicsXG5cdHdhcmVob3VzZTogJ2VtcHR5X3dhcmVob3VzZV8wMV8xay5oZHInLFxufTtcblxuZXhwb3J0IHR5cGUgTmd0c0Vudmlyb25tZW50UHJlc2V0cyA9IGtleW9mIHR5cGVvZiBFTlZJUk9OTUVOVF9QUkVTRVRTO1xuXG5jb25zdCBDVUJFTUFQX1JPT1QgPSAnaHR0cHM6Ly9yYXcuZ2l0aGFjay5jb20vcG1uZHJzL2RyZWktYXNzZXRzLzQ1NjA2MGEyNmJiZWI4ZmRmNzkzMjZmMjI0YjZkOTliOGJjY2U3MzYvaGRyaS8nO1xuXG5leHBvcnQgaW50ZXJmYWNlIE5ndHNJbmplY3RFbnZpcm9ubWVudE9wdGlvbnMge1xuXHRmaWxlczogc3RyaW5nIHwgc3RyaW5nW107XG5cdHBhdGg6IHN0cmluZztcblx0cHJlc2V0PzogTmd0c0Vudmlyb25tZW50UHJlc2V0cztcblx0ZXh0ZW5zaW9ucz86IChsb2FkZXI6IExvYWRlcikgPT4gdm9pZDtcblx0ZW5jb2Rpbmc/OiBUZXh0dXJlRW5jb2Rpbmc7XG59XG5cbmNvbnN0IGRlZmF1bHRGaWxlcyA9IFsnL3B4LnBuZycsICcvbngucG5nJywgJy9weS5wbmcnLCAnL255LnBuZycsICcvcHoucG5nJywgJy9uei5wbmcnXTtcblxuZXhwb3J0IGZ1bmN0aW9uIGluamVjdEVudmlyb25tZW50KFxuXHRvcHRpb25zOiAoKSA9PiBQYXJ0aWFsPE5ndHNJbmplY3RFbnZpcm9ubWVudE9wdGlvbnM+ID0gKCkgPT4gKHt9KSxcblx0eyBpbmplY3RvciB9OiB7IGluamVjdG9yPzogSW5qZWN0b3IgfSA9IHt9LFxuKSB7XG5cdHJldHVybiBhc3NlcnRJbmplY3RvcihpbmplY3RFbnZpcm9ubWVudCwgaW5qZWN0b3IsICgpID0+IHtcblx0XHRjb25zdCBhZGp1c3RlZE9wdGlvbnMgPSBjb21wdXRlZCgoKSA9PiB7XG5cdFx0XHRjb25zdCB7IHByZXNldCwgZXh0ZW5zaW9ucywgZW5jb2RpbmcsIC4uLnJlc3QgfSA9IG9wdGlvbnMoKTtcblx0XHRcdGxldCB7IGZpbGVzLCBwYXRoIH0gPSByZXN0O1xuXG5cdFx0XHRpZiAoZmlsZXMgPT0gbnVsbCkge1xuXHRcdFx0XHRmaWxlcyA9IGRlZmF1bHRGaWxlcztcblx0XHRcdH1cblxuXHRcdFx0aWYgKHBhdGggPT0gbnVsbCkge1xuXHRcdFx0XHRwYXRoID0gJyc7XG5cdFx0XHR9XG5cblx0XHRcdGlmIChwcmVzZXQpIHtcblx0XHRcdFx0dmFsaWRhdGVQcmVzZXQocHJlc2V0KTtcblx0XHRcdFx0ZmlsZXMgPSBFTlZJUk9OTUVOVF9QUkVTRVRTW3ByZXNldF07XG5cdFx0XHRcdHBhdGggPSBDVUJFTUFQX1JPT1Q7XG5cdFx0XHR9XG5cblx0XHRcdHJldHVybiB7IGZpbGVzLCBwcmVzZXQsIGVuY29kaW5nLCBwYXRoLCBleHRlbnNpb25zIH07XG5cdFx0fSk7XG5cblx0XHRjb25zdCBmaWxlcyA9IHBpY2soYWRqdXN0ZWRPcHRpb25zLCAnZmlsZXMnKTtcblx0XHRjb25zdCBtdWx0aUZpbGUgPSBjb21wdXRlZCgoKSA9PiBBcnJheS5pc0FycmF5KGZpbGVzKCkpKTtcblx0XHRjb25zdCByZXN1bHRPcHRpb25zID0gY29tcHV0ZWQoKCkgPT4gZ2V0RXh0ZW5zaW9uKGZpbGVzKCkpKTtcblx0XHRjb25zdCBleHRlbnNpb24gPSBwaWNrKHJlc3VsdE9wdGlvbnMsICdleHRlbnNpb24nKTtcblx0XHRjb25zdCBsb2FkZXIgPSBjb21wdXRlZCgoKSA9PiBnZXRMb2FkZXIoZXh0ZW5zaW9uKCkpKTtcblxuXHRcdGNvbnN0IHN0b3JlID0gaW5qZWN0U3RvcmUoKTtcblx0XHRjb25zdCBnbCA9IHN0b3JlLnNlbGVjdCgnZ2wnKTtcblxuXHRcdGNvbnN0IHRleHR1cmUgPSBzaWduYWw8VGV4dHVyZSB8IEN1YmVUZXh0dXJlIHwgbnVsbD4obnVsbCk7XG5cblx0XHRlZmZlY3QoKCkgPT4ge1xuXHRcdFx0Y29uc3QgW19leHRlbnNpb24sIF9tdWx0aUZpbGUsIF9maWxlc10gPSBbdW50cmFja2VkKGV4dGVuc2lvbiksIHVudHJhY2tlZChtdWx0aUZpbGUpLCBmaWxlcygpXTtcblxuXHRcdFx0aWYgKF9leHRlbnNpb24gIT09ICd3ZWJwJyAmJiBfZXh0ZW5zaW9uICE9PSAnanBnJyAmJiBfZXh0ZW5zaW9uICE9PSAnanBlZycpIHJldHVybjtcblxuXHRcdFx0Z2woKS5kb21FbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoXG5cdFx0XHRcdCd3ZWJnbGNvbnRleHRsb3N0Jyxcblx0XHRcdFx0KCkgPT4ge1xuXHRcdFx0XHRcdC8vIEB0cy1leHBlY3QtZXJyb3IgLSBmaWxlcyBpcyBjb3JyZWN0bHkgcGFzc2VkXG5cdFx0XHRcdFx0aW5qZWN0TG9hZGVyLmNsZWFyKG11bHRpRmlsZSA/IFtfZmlsZXNdIDogX2ZpbGVzKTtcblx0XHRcdFx0fSxcblx0XHRcdFx0eyBvbmNlOiB0cnVlIH0sXG5cdFx0XHQpO1xuXHRcdH0pO1xuXG5cdFx0Y29uc3QgcmVzdWx0ID0gaW5qZWN0TG9hZGVyKFxuXHRcdFx0bG9hZGVyLFxuXHRcdFx0Ly8gQHRzLWV4cGVjdC1lcnJvciAtIGVuc3VyZSB0aGUgZmlsZXMgaXMgYW4gYXJyYXlcblx0XHRcdCgpID0+IHtcblx0XHRcdFx0Y29uc3QgeyBmaWxlcyB9ID0gYWRqdXN0ZWRPcHRpb25zKCk7XG5cdFx0XHRcdHJldHVybiBBcnJheS5pc0FycmF5KGZpbGVzKSA/IFtmaWxlc10gOiBmaWxlcztcblx0XHRcdH0sXG5cdFx0XHR7XG5cdFx0XHRcdGV4dGVuc2lvbnM6IChsb2FkZXIpID0+IHtcblx0XHRcdFx0XHRjb25zdCB7IGV4dGVuc2lvbnMsIHBhdGggfSA9IGFkanVzdGVkT3B0aW9ucygpO1xuXHRcdFx0XHRcdGNvbnN0IHsgZXh0ZW5zaW9uIH0gPSByZXN1bHRPcHRpb25zKCk7XG5cdFx0XHRcdFx0aWYgKGV4dGVuc2lvbiA9PT0gJ3dlYnAnIHx8IGV4dGVuc2lvbiA9PT0gJ2pwZycgfHwgZXh0ZW5zaW9uID09PSAnanBlZycpIHtcblx0XHRcdFx0XHRcdC8vIEB0cy1leHBlY3QtZXJyb3IgLSBHYWlubWFwIHJlcXVpcmVzIGEgcmVuZGVyZXJcblx0XHRcdFx0XHRcdGxvYWRlci5zZXRSZW5kZXJlcihnbCgpKTtcblx0XHRcdFx0XHR9XG5cblx0XHRcdFx0XHRsb2FkZXIuc2V0UGF0aD8uKHBhdGgpO1xuXHRcdFx0XHRcdGlmIChleHRlbnNpb25zKSBleHRlbnNpb25zKGxvYWRlcik7XG5cdFx0XHRcdH0sXG5cdFx0XHR9LFxuXHRcdCk7XG5cblx0XHRlZmZlY3QoKCkgPT4ge1xuXHRcdFx0Y29uc3QgbG9hZGVyUmVzdWx0ID0gcmVzdWx0KCk7XG5cdFx0XHRpZiAoIWxvYWRlclJlc3VsdCkgcmV0dXJuO1xuXG5cdFx0XHR1bnRyYWNrZWQoKCkgPT4ge1xuXHRcdFx0XHRjb25zdCB7IGV4dGVuc2lvbiwgaXNDdWJlTWFwIH0gPSByZXN1bHRPcHRpb25zKCk7XG5cdFx0XHRcdGNvbnN0IF9tdWx0aUZpbGUgPSBtdWx0aUZpbGUoKTtcblx0XHRcdFx0Y29uc3QgeyBlbmNvZGluZyB9ID0gYWRqdXN0ZWRPcHRpb25zKCk7XG5cblx0XHRcdFx0Ly8gQHRzLWV4cGVjdC1lcnJvciAtIGVuc3VyZSB0ZXh0dXJlUmVzdWx0IGlzIGEgVGV4dHVyZSBvciBDdWJlVGV4dHVyZVxuXHRcdFx0XHRsZXQgdGV4dHVyZVJlc3VsdCA9IChfbXVsdGlGaWxlID8gbG9hZGVyUmVzdWx0WzBdIDogbG9hZGVyUmVzdWx0KSBhcyBUZXh0dXJlIHwgQ3ViZVRleHR1cmU7XG5cblx0XHRcdFx0Ly8gTk9URTogcmFjaW5nIGNvbmRpdGlvbiwgd2UgY2FuIHNraXAgdGhpc1xuXHRcdFx0XHQvLyAgd2UganVzdCBzYWlkIGFib3ZlIHRoYXQgaWYgbXVsdGlGaWxlIGlzIGZhbHNlLCBpdCBpcyBhIHNpbmdsZSBUZXh0dXJlXG5cdFx0XHRcdGlmICghX211bHRpRmlsZSAmJiBBcnJheS5pc0FycmF5KHRleHR1cmVSZXN1bHQpICYmIHRleHR1cmVSZXN1bHRbMF0gaW5zdGFuY2VvZiBDdWJlVGV4dHVyZSkge1xuXHRcdFx0XHRcdHJldHVybjtcblx0XHRcdFx0fVxuXG5cdFx0XHRcdGlmIChcblx0XHRcdFx0XHQhKHRleHR1cmVSZXN1bHQgaW5zdGFuY2VvZiBDdWJlVGV4dHVyZSkgJiZcblx0XHRcdFx0XHQoZXh0ZW5zaW9uID09PSAnanBnJyB8fCBleHRlbnNpb24gPT09ICdqcGVnJyB8fCBleHRlbnNpb24gPT09ICd3ZWJwJylcblx0XHRcdFx0KSB7XG5cdFx0XHRcdFx0dGV4dHVyZVJlc3VsdCA9ICh0ZXh0dXJlUmVzdWx0IGFzIGFueSkucmVuZGVyVGFyZ2V0Py50ZXh0dXJlO1xuXHRcdFx0XHR9XG5cblx0XHRcdFx0dGV4dHVyZVJlc3VsdC5tYXBwaW5nID0gaXNDdWJlTWFwID8gQ3ViZVJlZmxlY3Rpb25NYXBwaW5nIDogRXF1aXJlY3Rhbmd1bGFyUmVmbGVjdGlvbk1hcHBpbmc7XG5cblx0XHRcdFx0aWYgKCdjb2xvclNwYWNlJyBpbiB0ZXh0dXJlUmVzdWx0KVxuXHRcdFx0XHRcdCh0ZXh0dXJlUmVzdWx0IGFzIGFueSkuY29sb3JTcGFjZSA9IGVuY29kaW5nID8/IChpc0N1YmVNYXAgPyAnc3JnYicgOiAnc3JnYi1saW5lYXInKTtcblx0XHRcdFx0ZWxzZSAodGV4dHVyZVJlc3VsdCBhcyBhbnkpLmVuY29kaW5nID0gZW5jb2RpbmcgPz8gKGlzQ3ViZU1hcCA/IHNSR0JFbmNvZGluZyA6IExpbmVhckVuY29kaW5nKTtcblxuXHRcdFx0XHR0ZXh0dXJlLnNldCh0ZXh0dXJlUmVzdWx0KTtcblx0XHRcdH0pO1xuXHRcdH0pO1xuXG5cdFx0cmV0dXJuIHRleHR1cmUuYXNSZWFkb25seSgpO1xuXHR9KTtcbn1cblxuaW5qZWN0RW52aXJvbm1lbnQucHJlbG9hZCA9IChvcHRpb25zOiAoKSA9PiBQYXJ0aWFsPE5ndHNJbmplY3RFbnZpcm9ubWVudE9wdGlvbnM+ID0gKCkgPT4gKHt9KSkgPT4ge1xuXHRjb25zdCBfb3B0aW9ucyA9IG9wdGlvbnMoKTtcblx0bGV0IHsgZmlsZXMsIHBhdGggfSA9IF9vcHRpb25zO1xuXHRjb25zdCB7IHByZXNldCwgZXh0ZW5zaW9ucyB9ID0gX29wdGlvbnM7XG5cblx0aWYgKGZpbGVzID09IG51bGwpIHtcblx0XHRmaWxlcyA9IGRlZmF1bHRGaWxlcztcblx0fVxuXG5cdGlmIChwYXRoID09IG51bGwpIHtcblx0XHRwYXRoID0gJyc7XG5cdH1cblxuXHRpZiAocHJlc2V0KSB7XG5cdFx0dmFsaWRhdGVQcmVzZXQocHJlc2V0KTtcblx0XHRmaWxlcyA9IEVOVklST05NRU5UX1BSRVNFVFNbcHJlc2V0XTtcblx0XHRwYXRoID0gQ1VCRU1BUF9ST09UO1xuXHR9XG5cblx0Y29uc3QgeyBleHRlbnNpb24gfSA9IGdldEV4dGVuc2lvbihmaWxlcyk7XG5cblx0aWYgKGV4dGVuc2lvbiA9PT0gJ3dlYnAnIHx8IGV4dGVuc2lvbiA9PT0gJ2pwZycgfHwgZXh0ZW5zaW9uID09PSAnanBlZycpIHtcblx0XHR0aHJvdyBuZXcgRXJyb3IoJ2luamVjdEVudmlyb25tZW50OiBQcmVsb2FkaW5nIGdhaW5tYXBzIGlzIG5vdCBzdXBwb3J0ZWQnKTtcblx0fVxuXG5cdGNvbnN0IGxvYWRlciA9IGdldExvYWRlcihleHRlbnNpb24pO1xuXHRpZiAoIWxvYWRlcikgdGhyb3cgbmV3IEVycm9yKCdpbmplY3RFbnZpcm9ubWVudDogVW5yZWNvZ25pemVkIGZpbGUgZXh0ZW5zaW9uOiAnICsgZmlsZXMpO1xuXG5cdGluamVjdExvYWRlci5wcmVsb2FkKFxuXHRcdCgpID0+IGxvYWRlcixcblx0XHQvLyBAdHMtZXhwZWN0LWVycm9yIC0gZmlsZXMgaXMgY29ycmVjdGx5IHBhc3NlZFxuXHRcdCgpID0+IChBcnJheS5pc0FycmF5KGZpbGVzKSA/IFtmaWxlc10gOiBmaWxlcyksXG5cdFx0KGxvYWRlcikgPT4ge1xuXHRcdFx0bG9hZGVyLnNldFBhdGg/LihwYXRoKTtcblx0XHRcdGlmIChleHRlbnNpb25zKSBleHRlbnNpb25zKGxvYWRlcik7XG5cdFx0fSxcblx0KTtcbn07XG5cbmluamVjdEVudmlyb25tZW50LmNsZWFyID0gKGNsZWFyT3B0aW9uczogeyBmaWxlcz86IHN0cmluZyB8IHN0cmluZ1tdOyBwcmVzZXQ/OiBOZ3RzRW52aXJvbm1lbnRQcmVzZXRzIH0pID0+IHtcblx0Y29uc3Qgb3B0aW9ucyA9IHsgZmlsZXM6IGRlZmF1bHRGaWxlcywgLi4uY2xlYXJPcHRpb25zIH07XG5cdGxldCB7IGZpbGVzIH0gPSBvcHRpb25zO1xuXHRjb25zdCBwcmVzZXQgPSBvcHRpb25zLnByZXNldDtcblxuXHRpZiAocHJlc2V0KSB7XG5cdFx0dmFsaWRhdGVQcmVzZXQocHJlc2V0KTtcblx0XHRmaWxlcyA9IEVOVklST05NRU5UX1BSRVNFVFNbcHJlc2V0XTtcblx0fVxuXG5cdGluamVjdExvYWRlci5jbGVhcihmaWxlcyk7XG59O1xuXG5mdW5jdGlvbiB2YWxpZGF0ZVByZXNldChwcmVzZXQ6IHN0cmluZykge1xuXHRpZiAoIShwcmVzZXQgaW4gRU5WSVJPTk1FTlRfUFJFU0VUUykpXG5cdFx0dGhyb3cgbmV3IEVycm9yKCdQcmVzZXQgbXVzdCBiZSBvbmUgb2Y6ICcgKyBPYmplY3Qua2V5cyhFTlZJUk9OTUVOVF9QUkVTRVRTKS5qb2luKCcsICcpKTtcbn1cblxuZnVuY3Rpb24gZ2V0RXh0ZW5zaW9uKGZpbGVzOiBzdHJpbmcgfCBzdHJpbmdbXSkge1xuXHRjb25zdCBpc0N1YmVNYXAgPSBBcnJheS5pc0FycmF5KGZpbGVzKSAmJiBmaWxlcy5sZW5ndGggPT09IDY7XG5cdGNvbnN0IGlzR2Fpbm1hcCA9IEFycmF5LmlzQXJyYXkoZmlsZXMpICYmIGZpbGVzLmxlbmd0aCA9PT0gMyAmJiBmaWxlcy5zb21lKChmaWxlKSA9PiBmaWxlLmVuZHNXaXRoKCdqc29uJykpO1xuXHRjb25zdCBmaXJzdEVudHJ5ID0gQXJyYXkuaXNBcnJheShmaWxlcykgPyBmaWxlc1swXSA6IGZpbGVzO1xuXG5cdC8vIEV2ZXJ5dGhpbmcgZWxzZVxuXHRjb25zdCBleHRlbnNpb246IHN0cmluZyB8IGZhbHNlIHwgdW5kZWZpbmVkID0gaXNDdWJlTWFwXG5cdFx0PyAnY3ViZSdcblx0XHQ6IGlzR2Fpbm1hcFxuXHRcdFx0PyAnd2VicCdcblx0XHRcdDogZmlyc3RFbnRyeS5zdGFydHNXaXRoKCdkYXRhOmFwcGxpY2F0aW9uL2V4cicpXG5cdFx0XHRcdD8gJ2V4cidcblx0XHRcdFx0OiBmaXJzdEVudHJ5LnN0YXJ0c1dpdGgoJ2RhdGE6YXBwbGljYXRpb24vaGRyJylcblx0XHRcdFx0XHQ/ICdoZHInXG5cdFx0XHRcdFx0OiBmaXJzdEVudHJ5LnN0YXJ0c1dpdGgoJ2RhdGE6aW1hZ2UvanBlZycpXG5cdFx0XHRcdFx0XHQ/ICdqcGcnXG5cdFx0XHRcdFx0XHQ6IGZpcnN0RW50cnkuc3BsaXQoJy4nKS5wb3AoKT8uc3BsaXQoJz8nKT8uc2hpZnQoKT8udG9Mb3dlckNhc2UoKTtcblxuXHRyZXR1cm4geyBleHRlbnNpb24sIGlzQ3ViZU1hcCwgaXNHYWlubWFwIH07XG59XG5cbmZ1bmN0aW9uIGdldExvYWRlcihleHRlbnNpb246IHN0cmluZyB8IHVuZGVmaW5lZCkge1xuXHRjb25zdCBsb2FkZXIgPVxuXHRcdGV4dGVuc2lvbiA9PT0gJ2N1YmUnXG5cdFx0XHQ/IEN1YmVUZXh0dXJlTG9hZGVyXG5cdFx0XHQ6IGV4dGVuc2lvbiA9PT0gJ2hkcidcblx0XHRcdFx0PyBSR0JFTG9hZGVyXG5cdFx0XHRcdDogZXh0ZW5zaW9uID09PSAnZXhyJ1xuXHRcdFx0XHRcdD8gRVhSTG9hZGVyXG5cdFx0XHRcdFx0OiBleHRlbnNpb24gPT09ICdqcGcnIHx8IGV4dGVuc2lvbiA9PT0gJ2pwZWcnXG5cdFx0XHRcdFx0XHQ/IChIRFJKUEdMb2FkZXIgYXMgdW5rbm93biBhcyB0eXBlb2YgTG9hZGVyKVxuXHRcdFx0XHRcdFx0OiBleHRlbnNpb24gPT09ICd3ZWJwJ1xuXHRcdFx0XHRcdFx0XHQ/IChHYWluTWFwTG9hZGVyIGFzIHVua25vd24gYXMgdHlwZW9mIExvYWRlcilcblx0XHRcdFx0XHRcdFx0OiBudWxsO1xuXG5cdGlmICghbG9hZGVyKSB7XG5cdFx0dGhyb3cgbmV3IEVycm9yKCdpbmplY3RFbnZpcm9ubWVudDogVW5yZWNvZ25pemVkIGZpbGUgZXh0ZW5zaW9uOiAnICsgZXh0ZW5zaW9uKTtcblx0fVxuXG5cdHJldHVybiBsb2FkZXIgYXMgdHlwZW9mIExvYWRlcjtcbn1cbiJdfQ==
|