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.
Files changed (151) hide show
  1. package/abstractions/lib/catmull-rom-line.d.ts +139 -139
  2. package/abstractions/lib/edges.d.ts +164 -164
  3. package/abstractions/lib/grid.d.ts +39 -39
  4. package/abstractions/lib/line.d.ts +8 -7
  5. package/abstractions/lib/prism-geometry.d.ts +4 -4
  6. package/abstractions/lib/rounded-box.d.ts +3 -3
  7. package/abstractions/lib/text-3d.d.ts +46 -46
  8. package/abstractions/lib/text.d.ts +1 -2
  9. package/cameras/lib/cube-camera.d.ts +1 -1
  10. package/cameras/lib/orthographic-camera.d.ts +7 -8
  11. package/cameras/lib/perspective-camera.d.ts +2 -3
  12. package/controls/index.d.ts +9 -1
  13. package/controls/lib/camera-controls.d.ts +1 -1
  14. package/controls/lib/orbit-controls.d.ts +1 -1
  15. package/esm2022/abstractions/lib/billboard.mjs +3 -3
  16. package/esm2022/abstractions/lib/catmull-rom-line.mjs +3 -3
  17. package/esm2022/abstractions/lib/cubic-bezier-line.mjs +3 -3
  18. package/esm2022/abstractions/lib/edges.mjs +30 -34
  19. package/esm2022/abstractions/lib/gradient-texture.mjs +3 -3
  20. package/esm2022/abstractions/lib/grid.mjs +3 -3
  21. package/esm2022/abstractions/lib/helper.mjs +18 -20
  22. package/esm2022/abstractions/lib/line.mjs +25 -28
  23. package/esm2022/abstractions/lib/prism-geometry.mjs +3 -3
  24. package/esm2022/abstractions/lib/quadratic-bezier-line.mjs +3 -3
  25. package/esm2022/abstractions/lib/rounded-box.mjs +11 -17
  26. package/esm2022/abstractions/lib/text-3d.mjs +14 -17
  27. package/esm2022/abstractions/lib/text.mjs +11 -16
  28. package/esm2022/cameras/lib/camera-content.mjs +3 -3
  29. package/esm2022/cameras/lib/cube-camera.mjs +18 -27
  30. package/esm2022/cameras/lib/orthographic-camera.mjs +20 -20
  31. package/esm2022/cameras/lib/perspective-camera.mjs +21 -21
  32. package/esm2022/controls/index.mjs +6 -2
  33. package/esm2022/controls/lib/camera-controls.mjs +3 -3
  34. package/esm2022/controls/lib/orbit-controls.mjs +3 -3
  35. package/esm2022/controls/lib/scroll-controls.mjs +124 -130
  36. package/esm2022/gizmos/angular-three-soba-gizmos.mjs +5 -0
  37. package/esm2022/gizmos/index.mjs +6 -0
  38. package/esm2022/gizmos/lib/gizmo-helper/gizmo-helper.mjs +187 -0
  39. package/esm2022/gizmos/lib/gizmo-helper/gizmo-viewcube.mjs +304 -0
  40. package/esm2022/gizmos/lib/gizmo-helper/gizmo-viewport.mjs +332 -0
  41. package/esm2022/gizmos/lib/pivot-controls/axis-arrow.mjs +265 -0
  42. package/esm2022/gizmos/lib/pivot-controls/axis-rotator.mjs +264 -0
  43. package/esm2022/gizmos/lib/pivot-controls/pivot-controls.mjs +337 -0
  44. package/esm2022/gizmos/lib/pivot-controls/plane-slider.mjs +266 -0
  45. package/esm2022/gizmos/lib/pivot-controls/scaling-sphere.mjs +232 -0
  46. package/esm2022/gizmos/lib/transform-controls.mjs +177 -0
  47. package/esm2022/loaders/lib/loader.mjs +3 -3
  48. package/esm2022/materials/lib/custom-shader-material.mjs +8 -12
  49. package/esm2022/materials/lib/mesh-distort-material.mjs +3 -3
  50. package/esm2022/materials/lib/mesh-reflector-material.mjs +11 -15
  51. package/esm2022/materials/lib/mesh-refraction-material.mjs +26 -30
  52. package/esm2022/materials/lib/mesh-transmission-material.mjs +3 -3
  53. package/esm2022/materials/lib/mesh-wobble-material.mjs +3 -3
  54. package/esm2022/materials/lib/point-material.mjs +3 -3
  55. package/esm2022/misc/lib/bake-shadows.mjs +8 -10
  56. package/esm2022/misc/lib/computed-attribute.mjs +15 -19
  57. package/esm2022/misc/lib/decal.mjs +56 -53
  58. package/esm2022/misc/lib/fbo.mjs +10 -10
  59. package/esm2022/misc/lib/html/html-content.mjs +62 -67
  60. package/esm2022/misc/lib/html/html.mjs +3 -3
  61. package/esm2022/misc/lib/intersect.mjs +6 -9
  62. package/esm2022/misc/lib/sampler.mjs +48 -42
  63. package/esm2022/performances/lib/adaptive-dpr.mjs +29 -35
  64. package/esm2022/performances/lib/adaptive-events.mjs +12 -16
  65. package/esm2022/performances/lib/detailed.mjs +14 -18
  66. package/esm2022/performances/lib/instances/instances.mjs +16 -22
  67. package/esm2022/performances/lib/points/points.mjs +19 -25
  68. package/esm2022/performances/lib/segments/segments.mjs +11 -14
  69. package/esm2022/shaders/lib/mesh-refraction-material.mjs +1 -1
  70. package/esm2022/staging/index.mjs +3 -2
  71. package/esm2022/staging/lib/accumulative-shadows.mjs +19 -21
  72. package/esm2022/staging/lib/backdrop.mjs +19 -23
  73. package/esm2022/staging/lib/bb-anchor.mjs +3 -3
  74. package/esm2022/staging/lib/bounds.mjs +39 -43
  75. package/esm2022/staging/lib/camera-shake.mjs +14 -18
  76. package/esm2022/staging/lib/caustics.mjs +24 -28
  77. package/esm2022/staging/lib/center.mjs +28 -32
  78. package/esm2022/staging/lib/contact-shadows.mjs +3 -3
  79. package/esm2022/staging/lib/environment/environment.mjs +338 -0
  80. package/esm2022/staging/lib/environment/inject-environment.mjs +184 -0
  81. package/esm2022/staging/lib/float.mjs +3 -3
  82. package/esm2022/staging/lib/lightformer.mjs +18 -22
  83. package/esm2022/staging/lib/mask.mjs +8 -12
  84. package/esm2022/staging/lib/matcap-texture.mjs +9 -11
  85. package/esm2022/staging/lib/normal-texture.mjs +9 -11
  86. package/esm2022/staging/lib/randomized-lights.mjs +9 -13
  87. package/esm2022/staging/lib/render-texture.mjs +19 -20
  88. package/esm2022/staging/lib/sky.mjs +3 -3
  89. package/esm2022/staging/lib/spot-light.mjs +32 -40
  90. package/esm2022/staging/lib/stage.mjs +13 -17
  91. package/esm2022/stats/lib/stats.mjs +21 -27
  92. package/fesm2022/angular-three-soba-abstractions.mjs +119 -138
  93. package/fesm2022/angular-three-soba-abstractions.mjs.map +1 -1
  94. package/fesm2022/angular-three-soba-cameras.mjs +57 -64
  95. package/fesm2022/angular-three-soba-cameras.mjs.map +1 -1
  96. package/fesm2022/angular-three-soba-controls.mjs +141 -1470
  97. package/fesm2022/angular-three-soba-controls.mjs.map +1 -1
  98. package/fesm2022/angular-three-soba-gizmos.mjs +2318 -0
  99. package/fesm2022/angular-three-soba-gizmos.mjs.map +1 -0
  100. package/fesm2022/angular-three-soba-loaders.mjs +3 -3
  101. package/fesm2022/angular-three-soba-materials.mjs +52 -62
  102. package/fesm2022/angular-three-soba-materials.mjs.map +1 -1
  103. package/fesm2022/angular-three-soba-misc.mjs +194 -195
  104. package/fesm2022/angular-three-soba-misc.mjs.map +1 -1
  105. package/fesm2022/angular-three-soba-performances.mjs +90 -114
  106. package/fesm2022/angular-three-soba-performances.mjs.map +1 -1
  107. package/fesm2022/angular-three-soba-shaders.mjs.map +1 -1
  108. package/fesm2022/angular-three-soba-staging.mjs +427 -430
  109. package/fesm2022/angular-three-soba-staging.mjs.map +1 -1
  110. package/fesm2022/angular-three-soba-stats.mjs +20 -26
  111. package/fesm2022/angular-three-soba-stats.mjs.map +1 -1
  112. package/gizmos/README.md +3 -0
  113. package/gizmos/index.d.ts +5 -0
  114. package/gizmos/lib/gizmo-helper/gizmo-helper.d.ts +45 -0
  115. package/gizmos/lib/gizmo-helper/gizmo-viewcube.d.ts +74 -0
  116. package/gizmos/lib/gizmo-helper/gizmo-viewport.d.ts +64 -0
  117. package/{controls → gizmos}/lib/pivot-controls/pivot-controls.d.ts +1 -1
  118. package/gizmos/lib/transform-controls.d.ts +50 -0
  119. package/materials/lib/mesh-transmission-material.d.ts +1 -1
  120. package/misc/lib/computed-attribute.d.ts +2 -2
  121. package/misc/lib/decal.d.ts +4 -4
  122. package/misc/lib/html/html-content.d.ts +6 -6
  123. package/misc/lib/html/html.d.ts +1 -1
  124. package/misc/lib/sampler.d.ts +1 -3
  125. package/package.json +14 -8
  126. package/performances/lib/instances/instances.d.ts +3 -3
  127. package/shaders/lib/mesh-refraction-material.d.ts +2 -5
  128. package/staging/index.d.ts +2 -1
  129. package/staging/lib/camera-shake.d.ts +0 -1
  130. package/staging/lib/caustics.d.ts +1 -1
  131. package/staging/lib/center.d.ts +2 -2
  132. package/staging/lib/contact-shadows.d.ts +1 -1
  133. package/staging/lib/{environment.d.ts → environment/environment.d.ts} +20 -49
  134. package/staging/lib/environment/inject-environment.d.ts +33 -0
  135. package/staging/lib/lightformer.d.ts +1 -1
  136. package/staging/lib/mask.d.ts +1 -1
  137. package/staging/lib/matcap-texture.d.ts +0 -1
  138. package/staging/lib/normal-texture.d.ts +0 -1
  139. package/staging/lib/render-texture.d.ts +1 -1
  140. package/staging/lib/sky.d.ts +1 -1
  141. package/staging/lib/stage.d.ts +3 -2
  142. package/esm2022/controls/lib/pivot-controls/axis-arrow.mjs +0 -263
  143. package/esm2022/controls/lib/pivot-controls/axis-rotator.mjs +0 -264
  144. package/esm2022/controls/lib/pivot-controls/pivot-controls.mjs +0 -340
  145. package/esm2022/controls/lib/pivot-controls/plane-slider.mjs +0 -266
  146. package/esm2022/controls/lib/pivot-controls/scaling-sphere.mjs +0 -232
  147. package/esm2022/staging/lib/environment.mjs +0 -481
  148. /package/{controls → gizmos}/lib/pivot-controls/axis-arrow.d.ts +0 -0
  149. /package/{controls → gizmos}/lib/pivot-controls/axis-rotator.d.ts +0 -0
  150. /package/{controls → gizmos}/lib/pivot-controls/plane-slider.d.ts +0 -0
  151. /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==